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PHP 是 一 种 易于 学 习 和 使 用 的 后 台 开 发 技术 ,用 户 只 需 具备 很 少 的 编程 知识 ， 就 可 以 使 用 PHP 
建立 一 个 具有 交互 功能 的 Web 站 点 。PHP 同样 也 是 一 种 嵌入 式 HTML 脚本 语言 ， 大 多 数 语法 来 源 
于 C 语言 ， 也 有 一 部 分 PHP 特性 借鉴 于 Java 和 Perl 语言 。 

PHP 最 大 的 特色 是 简单 并 与 MySQL 天 生 的 结合 性 。 从 目前 市 场 情况 看 ,大约 有 2200 万 家 网 站 
采用 PHP 技术 ,而 且 数量 还 在 持续 增加 中 .PHP 技术 也 受到 计算 机 工业 巨头 的 支持 , 像 IBM 和 Oracle 
都 致力 于 开发 支持 PHP 软件 以 顺利 读 取 其 下 数据 库 的 程序 ， 支 持 PHP 的 发 展 。 

常用 的 动态 Web 技术 有 JSP、ASP、PHP 等 ， 这 些 技术 各 有 优 缺 点 ，PHP 技术 具有 实用 性 好 、 
功能 强大 、 成 本 较 低 等 特性 ， 对 于 个 人 用 户 来 说 ， 学 习 和 使 用 PHP 技术 是 一 个 很 好 的 选择 。 

全 书 共 分 为 3 篇 21 章 ， 第 1 章 到 第 8 章 为 “入 门 篇 ” 第 9 章 到 第 15 章 为 “提高 编 ” 第 16 
章 到 第 21 章 为 “实践 篇 ” 以 “入 门 篇 一 提高 篇 一 实践 篇 ”为 线索 具体 展开 ， 涵 盖 了 PHP 网 络 编程 
各 方面 的 知识 。 本 书 从 PHP 基础 入 手 ， 简 单 介绍 PHP 的 运行 环境 配置 、 语 法 、 函 数 等 知识 。 然 后 
全 面 详细 地 介绍 PHP 的 高 级 知识 ， 如 文件 操作 、MySQL 数据 库 、 访 问 MySQL 数据 库 、Cookie、 
XML、PEAR， 为 开发 比较 复杂 的 网 站 打下 坚实 的 基础 。 最 后 以 Web 开发 中 常见 的 6 种 典型 案例 ， 
演示 使 用 PHP 开发 和 部 署 网 站 的 过 程 ， 如 聊天 室 、 留 言 板 等 。 

本 书 全 面 介绍 PHP 面向 实践 的 开发 和 应 用 知识 ， 具 有 如 下 特点 : 

。 内 容 全 面 本 书 是 一 本 大 全 性 质 的 PHP 编程 图 书 ， 突 出 介绍 PHP 面向 实际 的 开发 知识 。 读 
者 学 习 本 书 之 后 ， 可 以 全 面 掌 握 PHP 动态 网 站 的 开发 实践 知识 。 
实例 丰富 ”全书 每 个 知识 点 的 讲解 都 配 有 大 量 可 实际 运行 的 实例 ， 读 者 可 以 边 学 习 边 实践 ， 
快速 、 全 面 地 掌握 PHP 的 开发 方法 和 技巧 。 书 中 最 后 一 篇 还 提供 了 6 个 典型 开发 案例 ， 履 
盖 了 PHP 开发 和 部 署 网 站 的 全 部 过 程 。 

本 书 配套 光盘 提供 了 本 书 源 代码 ( 包括 6 个 完整 的 应 用 程序 )、PHP 软件 配置 ， 以 及 教学 视 
频 。 读 者 只 要 按照 书 中 的 案例 上 机 练习 、 举 一 反 三 ， 就 可 以 根据 自己 的 需要 开发 出 功能 强大 
的 Web 动态 网 站 。 

本 书 适 合 于 中 、 高 级 PHP 网 站 开发 人 员 ， 特 别 适合 于 有 编程 基础 ， 希 望 全 面 学 习 PHP 技术 ， 
提高 实际 应 用 能 力 的 读者 群体 。 

除了 封面 署名 人 员 之 外 ， 参 与 本 书 编 写 的 人 员 还 有 李 乃 文 、 张 仁 画 、 夏 小 军 、 赵 振江 、 李 振 山 、 
李 文 采 、 吴 越 胜 、 李 海 庆 、 何 永国 、 李 海峰 、 陶 丽 、 吴 俊 海 、 安 征 、 张 岗 屹 、 崔 群 法 、 王 咏 梅 、 康 
显 丽 、 辛 爱 军 、 牛 小 平 、 贾 栓 稳 、 王 立新 、 苏 静 、 赵 元 庆 、 郭 磊 、 徐 铭 、 李 大 庆 、 王 蕾 、 张 勇 、 孝 
安 林 等 。 在 编写 过 程 中 难免 会 有 疏 漏 ， 欢 迎 读者 与 我 们 联系 ， 帮 助 我 们 改正 提高 。 
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/学 习 目 标 |owiecwve 


构建 动态 网 站 ， 可 以 选择 多 种 动态 网 站 开发 技术 ， 如 JSP、PHP、ASP、ASPNET 等 。 在 如 此 
众多 的 动态 网 站 开发 技术 中 ，PHP 是 一 种 易于 学 习 和 使 用 的 后 台 开发 技术 。 用 户 只 需要 具备 很 少 的 
编程 知识 ， 就 可 以 使 用 PHP 建立 具有 交互 功能 的 Web 站 点 。PHP 同样 也 是 一 种 嵌入 式 HTML 脚本 
语言 。PHP 脚本 语言 的 大 多 数 语法 来 源 于 C 语言 ， 也 有 一 部 分 PHP 特性 借鉴 于 Java 和 Perl。 这 种 
语言 的 目的 是 让 Web 开发 人 员 能 够 快速 高 效 地 写 出 动态 生成 的 页 面 代码 。 

本 章 将 简要 介绍 PHP 现 有 的 版 本 ， 并 重点 介绍 构建 PHP 的 运行 平台 ， 即 Apache 和 PHP 的 安 
装 及 配置 。 本 章 给 出 了 一 个 案例 ， 测 试 PHP 运行 平台 。 


A 
内 容 摘要 | avstrsct 


了 解 PHP 产生 的 原因 

了 解 PHP 发 展 的 历程 和 版 本 

掌握 PHP 程序 的 特点 

掌握 获取 及 安装 Apache 和 PHP 软件 
熟练 掌握 PHP 运行 环境 的 配置 

掌握 PHP 环境 的 测试 和 定制 

掌握 处 理 PHP 的 错误 

了 解 PHP 的 配置 指令 

了 解 PHP 开发 的 流程 
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PHP 的 产生 伴随 着 不 断 的 改进 ,目前 已 经 成 为 动态 网 页 开发 技术 的 主流 技术 之 一 。 本 节 介 绍 PHP 
的 现 有 版 本 ， 为 后 面 深入 学 习 PHP 知识 打下 基础 。 


1.1.1 PHP4.0 和 PHP 5.0 


PHP 的 发 展 经 历 了 两 个 比较 重要 的 阶段 一 一 PHP 4.0 版 本 和 PHP 5.0 版 本 。 本 节 从 两 个 版 本 的 产 
生 入 手 ， 详 细 介绍 各 版 本 不 同 的 功能 ， 以 及 新 的 版 本 相对 于 旧版 本 的 改进 。 
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1. PHP 4.0 简介 


1998 年 的 冬天 , PHP 3.0 版 本 发 布 不 久 , Andi Gutmans 和 Zeev Suraski 开始 重新 编写 PHP 代码 。 
设计 目标 是 增强 复杂 程序 运行 时 的 性 能 和 PHP 自身 代码 的 模块 性 。 

基于 Zend Engine 引擎 并 结合 了 更 多 新 功能 的 PHP 4.0， 于 2000 年 5 月 发 布 了 官方 正式 版 本 。 
除了 更 高 的 性 能 以 外 ，PHP 4.0 还 包含 了 其 他 一 些 关键 功能 ， 比 如 ， 支 持 更 多 的 Web 服务 器 、HTTP 
Sessions 支持 、 输 出 缓冲 、 更 安全 的 处 理 用 户 输入 的 方法 及 一 些 新 的 语言 结构 。PHP 的 开发 小 组 有 
很 多 优秀 的 开发 人 员 , 同时 还 有 大 量 的 优秀 人 才 在 进行 PHP 相关 工程 的 开发 工作 , 如 PEAR 和 PHP 


文档 的 工程 。 


PHP 4.0 与 以 前 的 版 本 相 比较 有 以 下 的 特点 ， 如 表 1-1 所 示 。 


特 点 
改进 了 资源 处 理 


面向 对 象 的 支持 


表 1-1 PHP 4.0 特点 说 明 
说 了 明 

可 扩展 性 是 PHP 4.0 的 主要 缺点 之 一 。 这 主要 是 因为 设计 者 低估 了 这 种 语言 ， 没 考虑 
到 它 会 大 量 用 于 大 规模 应 用 程序 。 这 就 使 得 开发 人 员 开始 重新 考虑 这 种 语言 的 机 制 。 
最 终 促使 在 PHP 4.0 中 对 资源 处 理 进行 了 大 幅 改 进 
PHP 4.0 在 一 定 程度 上 结合 了 面向 对 象 的 功能 ， 尽 管 一 般 认 为 这 只 是 一 个 很 平常 的 实 
现 。 但 是 ， 对 于 使 用 传统 面向 对 象 程序 设计 语言 的 用 户 来 说 ， 这 个 新 特性 在 吸引 这 些 
用 户 方面 起 到 了 非常 重要 的 作用 。 除 了 对 象 重 载 和 运行 时 类 信息 外 ，PHP 还 支持 标准 
的 类 和 对 象 开发 方法 。 


内 置 的 会 话 处 理 支持 
加 密 


ISAPI 支 持 


在 PHP 4.0 中 HTTP 会 话 处 理 则 是 内 置 的 功能 。 这 个 特性 使 得 开发 人 员 可 以 高 效 轻松 
地 跟踪 用 户 活动 和 喜好 

MCrypt 《http://merypt.sourceforge.net) 库 引入 默认 发 行 包 中 ， 为 用 户 提供 了 完全 加 密 
和 散 列 加 密 ， 使 用 的 加 密 算法 包括 Blowfish、MD5、SHA1 和 三 重 DES 等 
使 用 户 能 够 将 PHP 与 微软 的 TS Web 服务 器 〈 作 为 一 个 ISAPI 模块 ) 结合 使 用 ， 大 大 


提高 了 性 能 和 安全 性 
内 置 COM/DCOM 支持 | 对 Windows 用 户 来 说 ， 另 一 个 好 处 是 PHP 4.0 能 够 访问 和 实例 化 COM 对 象 。 这 项 功 


内 置 Java 支持 
与 Perl 兼容 的 正则 
表达 式 (PCRE) 库 


能 扩展 了 与 Windows 应 用 程序 的 互 操作 性 

这 也 是 PHP 在 互 操作 性 方面 的 一 大 进步 ， PHP 4.0 支持 PHP 应 用 程序 绑 定 Java 对 象 

Perl 语言 在 字符 串 解析 领域 一 直 以 来 雄霸 天 下 ， 占 据 着 统治 地 位 。 开 发 人 员 知 道 ， 如 
果 想 让 PHP 得 到 广泛 认可 ,强大 的 正则 表达 式 功能 会 起 到 重要 的 作用 。 他 们 的 做 法 只 
是 集成 Perl 的 功能 , 而 不 是 重新 开发 , 并 将 PCRE 库 的 包 集成 在 PHP 的 默认 发 行 包 中 
(PHP 4.2.0) 


除了 表 中 所 列举 的 这 些 特 性 外 ，PHP 4.0 还 添加 了 几 百 项 功能 ， 大 大 提升 了 这 种 语言 的 能 力 。 
本 书 将 讨论 其 中 大 部 分 功能 ， 因 为 这 些 功 能 在 PHP 5.0 中 仍然 很 重要 。 

PHP 4.0 和 其 整合 的 Zend 引擎 极 大 地 增强 了 PHP 的 性 能 和 兼容 性 , 对 细节 代码 也 十 分 注意 , 所 
以 从 PHP 3.0 到 PHP 4.0 的 移植 要 比 从 PHP/FI 2.0 到 PHP 3.0 的 移植 容易 得 多 。 很 多 PHP 3.0 的 代码 
无 须 修改 就 可 以 在 PHP 4.0 中 运行 ， 但 是 需要 在 转换 程序 运行 环境 时 注意 一 些 细节 。 


2. PHP 5.0 简介 


2004 年 7 月 13 日 , PHP 5.0 发布 , 该 版 本 以 Zend 引擎 I 为 引擎 , 并 且 加 入 了 新 的 功能 , 如 PHP 
Data Objects (PDO)。 目 前 PHP 最 新 的 版 本 是 2007 年 6 月 1 号 发 布 的 PHP 5.2.3 版 本 。 

PHP 5.0 是 PHP 语言 发 展 历程 中 的 另 一 座 分 水 岭 。 虽 然 前 面 的 主要 版 本 已 经 增加 了 许多 库 ， 但 
PHP 5.0 在 现 有 的 功能 上 又 进行 了 许多 改进 ， 并 且 增 加 了 只 有 成 熟 的 编程 语言 体系 结构 才 有 的 一 些 
特性 ， 其 详细 信息 如 表 1-2 所 示 。 


表 1-2 特性 改进 

新 增 特 性 说 了 明 
极 大 地 提高 了 面向 对 象 能 力 ”| PHP 的 面向 对 象 体系 结构 得 到 了 改进 ， 这 是 PHP 5.0 最 突出 的 特点 。PHP 5.0 增 
加 了 很 多 功能 ， 如 显 式 构造 函数 和 析 构 函数 、 对 象 克隆 、 类 抽象 、 变 量 作用 域 
和 接口 等 
PHP 5.0 开始 支持 异常 处 理 。 在 许多 语言 中 ， 如 C++、C#、Python 和 Java 等 ， 
异常 处 理 一 直 都 是 错误 管理 方面 的 中 流 碟 柱 ， 它 为 建立 标准 化 的 错误 报告 逻辑 
提供 了 一 种 绝 佳 的 方法 
之 前 版 本 的 PHP 默认 地 将 字符 串 看 作 数 组 , 这 也 反映 了 PHP 原 有 的 数据 类 型 观 
点 不 够 严密 。 这 种 策略 在 PHP 5.0 中 有 所 调整 , 引入 了 一 种 专门 的 字符 串 偏 移 量 
(offset) 语法 ， 而 以 前 的 方法 已 经 废弃 不 用 
现在 的 XML 支持 建立 在 libxml2 库 基 础 上 ， 并 引入 了 一 个 很 新 且 非 常 有 前 途 的 
扩展 包 来 解析 和 处 理 XML: SimpleXML。 此 外 ，PHP 5.0 还 支持 SOAP 扩展 
PHP 5.0 为 功能 强大 且 简洁 的 SQLite 数据 库 服务 器 (http://www.sqlite.org) 提供 
了 支持 。 如 果 开 发 人 员 需 要 使 用 一 些 只 有 重量 级 数据 库 产品 中 才 有 的 特性 ， 同 
时 不 希望 带 来 相应 的 管理 开销 ，SQLite 则 会 是 一 个 方便 的 解决 方案 


try/catch 异常 处 理 


字符 串 处 理 


改进 的 XML 和 Web 服务 支持 


对 SQLite 的 内 置 支持 


1.1.2 PHP 的 特性 


使 用 PHP 有 许多 好 处 ， 如 实用 性 、 强 大 的 功能 、 成 本 较 低 等 。 对 于 个 人 来 说 学 习 和 使 用 PHP 
是 一 个 很 好 的 选择 。 虽 然 PHP 是 开放 源码 项 目 ， 没 有 什么 商业 支持 ， 并 且 执行 速度 缓慢 〈 直 到 PHP 
4.0 之 前 )， 但 是 PHP 的 邮件 列表 很 有 用 ， 而 且 除 非 正在 运行 像 Yahoo! 或 者 Amazon.com 这 样 的 极 
受 欢迎 的 站 点 ， 不 会 感觉 出 PHP 的 速度 与 其 他 站 点 的 运行 速度 有 什么 不 同 。 下 面 详细 介绍 PHP 5.0 
有 具 有 的 优点 。 

1. 实用 性 

从 PHP 的 发 展 历史 可 以 知道 , PHP 的 产生 是 为 了 解决 特定 的 现实 问题 , 而 不 是 为 了 设计 一 种 新 
的 语言 ， 用 来 解决 固定 的 假设 的 问题 。 在 PHP 的 发 展 道路 上 ， 不 是 明确 地 要 增加 PHP 的 功能 ， 而 
是 为 了 解决 用 户 的 具体 需求 而 增加 功能 。 这 样 做 的 结果 是 建立 一 种 入 门 非常 容易 , 对 用 户 需 求 较 低 ， 
不 需要 用 户 具 备 深厚 的 计算 机 知识 ， 语 法 基础 比较 简单 的 语言 。 

对 于 刚 入 门 的 用 户 来 说 ， 一 个 有 用 的 PHP 脚本 可 能 只 包含 一 行 代码 ， 与 C 语言 不 同 ， 它 不 需 
要 导入 库 函 数 。 例如， 下 面 的 代码 就 是 一 个 完整 的 PHP 脚本 , 其 目的 是 以 类 似 于 September 23, 2005 
的 格式 输出 当前 的 日 期 : 


<?php echo date("F j,y");?> 


PHP 语言 强调 紧凑 性 ， 这 反映 在 它 能 嵌 套 函数 。 例 如 ， 通 过 在 一 行 代码 中 按 特定 的 顺序 调用 函 
数 , 可 以 对 一 个 值 进行 一 系列 修改 。 下 面 的 例子 将 生成 一 个 由 5 个 字母 或 数字 字符 组 成 的 伪 随 机 串 ， 
如 a3jh8: 


SrandomString=substr (mds (microtime()),0,5)7 


PHP 是 一 种 弱 类 型 的 语言 ， 即 类 型 松散 的 语言 ， 这 意味 着 不 需要 明确 地 创建 变量 、 指 派 类 型 或 
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撤销 变量 ， 也 没有 绝对 禁止 进行 这 些 操作 。PHP 在 内 部 处 理 这 些 情 况 ， 脚 本 中 使 用 变量 时 PHP 会 动 
态 创建 变量 ， 并 使 用 最 优 推测 规则 自动 指派 变量 的 类 型 。 例 如 ，PHP 认为 下 面 的 一 组 语句 是 完全 合 
法 的 : 

S$string value=" 你 好 "7 

echo ($string value) 7 

Sradius=2.07 


$pi=3.14159; 
$area=$pi*$radius*#$radius; 


PHP 会 在 脚本 结束 时 自动 撤销 变量 , 将 资源 返回 给 系统 。 从 这 些 方面 来 看 ， 由 于 PHP 在 内 部 处 
理 了 编程 的 许多 管理 方面 的 问题 ， 这 就 允许 开发 人 员 集 中 精力 完成 最 终 的 目标 ， 也 就 是 开发 一 个 实 
用 的 应 用 程序 。 

2. 强大 功能 

在 前 面 介绍 PHP 5.0 时 就 已 经 提 到 ， 这 个 新 版 本 相对 于 以 前 的 版 本 更 重视 质量 ， 而 非 数 量 。 以 
前 的 主要 版 本 向 PHP 的 默认 库 增加 很 多 特性 ， 每 次 发 行 新 版 本 都 会 增加 几 百 项 新 功能 。 目 前 ，PHP 
有 113 个 可 用 的 库 ， 共 有 1000 余 项 功能 ， 主 要 功能 如 下 所 示 : 

。 创建 并 处 理 Macromedia Flash、 图 片 和 PDF 文件 。 

将 密码 与 字典 数据 和 容易 破解 的 模式 进行 比较 ， 评 估 密 码 的 可 猜测 性 。 

与 轻 量 级 目录 访问 协议 (LDAP ) 通信 。 

使 用 基于 POSIX 和 Perl 的 正则 表达 式 库 解 析 最 复杂 的 字符 串 。 

通过 存储 在 纯 文 本 文件 、 数 据 库 或 Microsoft 活动 目录 中 的 登录 凭证 来 鉴别 用 户 身份 。 
与 多 种 协议 通信 ， 包 括 IMAP、POP3、NNTP 和 DNS 等 。 

与 大 量 信 用 卡 处 理解 决 方案 通信 。 

除了 上 面 这 些 新 增加 的 功能 特性 之 外 ， 还 具有 其 他 的 新 功能 。 在 后 面 的 章节 中 会 陆续 讲 到 。 

3. 可 选择 性 

PHP 开发 人 员 很 少 只 局 限于 一 种 实现 方案 。 相 反 ， 这 种 语言 为 用 户 提供 了 充分 的 选择 。 例 如 ， 
PHP 对 数据 库 的 支持 。PHP 为 不 少 于 25 种 数据 库 产 品 提供 了 内 置 支持 ， 包 括 Adabas D、dBase、 
Empress、FilePro、FrontBase、Hyperwave、 IBM DB2、Informix、Ingres、InterBase、mSQL、Direct 
MS-SQL、 MySQL、Oracle、Ovrimos、PostgreSQL、Solid、Sybase、UNIX dbm 和 Velocis。 此 外 ， 
也 可 以 利用 抽象 层 功能 来 访问 Berkeley DB 类 型 的 数据 库 , 还 有 两 个 数据 库 抽象 层 可 用 , 一 个 是 dbx 
模块 ， 另 一 个 是 通过 PEAR 的 PEAR DB。 

PHP 强大 的 字符 串 解析 功能 也 为 用 户 提供 了 丰富 的 可 选择 性 。 除 了 超过 85 个 字符 串 处 理 函 数 
之 外 ，PHP 还 支持 基于 POSIX 和 Perl 的 正则 表达 式 格式 。 这 种 灵活 性 使 不 同 水 平 的 用 户 都 能 获 益 ， 
不 仅 能 够 〈 利 用 字符 串 处 理 函 数 ) 立即 完成 复杂 的 字符 串 操作 ， 还 可 以 〈 利 用 正则 表达 式 ) 将 有 类 
似 功能 的 程序 〈 如 Perl 和 Python) 快速 移植 到 PHP。 

无 论 是 函数 式 编程 语言 ， 还 是 面向 对 象 程序 设计 〈ObjectOriented Programming，OOP) 语言 ， 
PHP 对 二 者 都 提供 了 全 面 的 支持 。 虽 然 PHP 最 初 只 是 一 种 函数 式 语言 , 但 开发 人 员 很 快 就 意识 到 提 
供 流行 的 OOP 范 型 的 重要 性 ， 并 开始 实现 一 种 可 扩展 的 解决 方案 。 

这 里 反复 强调 的 重点 是 ，PHP 允许 充分 利用 目前 掌握 的 技能 ， 只 需 投 入 很 少 的 时 间 就 能 很 快 地 
开始 PHP 开发 。 这 种 策略 在 整个 语言 中 频频 出 现 ， 这 里 提 到 的 只 是 其 中 很 少 的 一 部 分 例子 。 


4. 成 本 

PHP 从 一 开始 就 对 使 用 、 修 改 和 再 分 发 没有 任何 限制 。 最 近 几 年 ， 满 足 这 种 开放 许可 限制 的 软 
件 称 为 开源 软件 “Open-Source Software，OSS)。 开 源 软 件 和 因特网 就 像 面包 和 黄油 一 样 密 不 可 分 。 
开源 项 目 如 Sendmail、Bind、Linux 和 Apache 都 在 因特网 的 发 展 方面 起 到 非常 重要 的 作用 。 虽 然 开 
源 软件 最 大 优势 是 可 以 自由 使 用 ， 但 它 还 有 另外 几 个 同样 重要 的 特点 《甚至 更 重要 )。 

。 没有 大 多 数 商业 产品 所 要 求 的 许可 限制 ”商业 软件 往往 有 许多 许可 限制 ,而 开源 软件 的 用 户 
没有 这 些 限制 。 虽 然 在 许可 权限 上 存在 差异 ， 但 一 般 来 讲 ， 用 户 都 能 自由 地 修改 和 重新 分 发 
开源 软件 ， 还 能 将 开源 软件 整合 到 其 他 产品 中 。 

。 开放 式 开 发 和 审计 过 程 ”虽然 也 曾 有 过 一 些 意外 事件 , 但 开源 软件 在 安全 方面 还 是 享有 很 好 
的 声誉 。 这 种 高 标准 正 是 开放 式 开发 和 审计 过 程 的 结果 。 因 为 任何 人 都 能 自由 使 用 源 代码 ， 
所 以 安全 漏洞 和 潜在 的 问题 会 很 快 被 发 现 并 得 以 修复 。 开 源 倡导 者 Eric S. Raymond 很 好 地 
总 结 了 这 项 优点 ， 他 指出 : “只 要 有 足够 的 眼睛 ， 所 有 的 bug 都 无 处 通 形 ”。 

。 鼓励 参与 ”开发 团队 不 限于 某 个 组 织 。 任 何 感 兴趣 的 人 ， 只 要 有 具有 相应 的 能 力 ， 都 可 以 自由 
地 加 入 项 目 中 。 由 于 不 对 成 员 进 行 限制 ， 这 就 大 大 增加 了 项 目的 人 才 储 备 ， 必 然 能 贡献 出 更 
高 质量 的 产品 。 


1.1.3 ”PHP 的 环境 需求 


PHP 技术 是 一 种 动态 网 站 开发 技术 。 可 以 使 用 PHP 构建 网 站 ， 与 客户 端 进行 动态 交互 。PHP 
的 建立 及 运行 需要 相应 的 环境 。 在 安装 PHP 作为 WWW 服务 器 的 一 部 分 时 ， 可 以 使 用 Linux 操作 
系统 ， 或 者 是 Windows NT/2000 等 Win32 API 的 平台 。 

当然 ， 大 部 分 人 都 会 使 用 Linux 来 当 作 PHP 的 执行 平台 。 实 际 上 ，Linux+Apache+PHP 应 是 最 
经 济 的 选择 ， 因 为 这 样 的 组 合 几乎 是 不 用 花 钱 的 ， 成 本 与 效益 比 也 是 最 好 的 。 许 多 成 功 网 站 都 是 采 
用 这 种 组 合 。Linux 操作 系统 方面 ， 可 以 选择 各 式 的 Linux 套件 ， 包 括 Slackware Linux、RedHat、 
OpenLinux、SuSE… 等 ， 可 以 购买 或 者 在 各 大 FTP 网 站 下 载 完 整 的 系统 安装 文件 。 
Windows 操作 系统 系列 的 用 户 大 多 数 都 会 选择 Apache+ PHP， 或 者 IS+PHP。 在 Windows 下 使 
用 PHP， 不 需要 安装 其 他 的 组 件 ， 只 需要 把 PHP 需要 的 软件 安装 就 可 以 了 ， 如 服务 器 Apache、 脚 
本 语言 分 析 PHP 动态 库 、 相 应 的 PHP 页 面 的 开发 工具 。 

Apache 服务 器 是 目前 最 多 WWW 网 站 采用 的 服务 器 , 可 以 在 网 站 http://www.apache.org 下 载 最 
新 版 本 的 程序 及 相关 文件 ， 也 可 以 用 它 的 Mirror 网 站 下 载 。PHP 则 可 以 在 它 的 官方 网 站 
http://www.php.net 下 载 所 需要 的 程序 。 


1.1.4 ”PHP 的 数据 库 集成 功能 


构建 一 个 动态 网 站 ， 数 据 的 交互 是 不 可 避免 的 。 我 们 知道 ， 数 据 是 程序 操作 的 对 象 ， 而 数据 库 
管理 系统 为 使 用 数据 和 存储 数据 提供 了 最 复杂 、 最 强大 的 功能 支持 。 因 此 ，PHP 成 熟 的 标志 之 一 ， 
就 是 和 数据 库 联系 的 程度 。 和 PHP 结合 使 用 的 数据 库 有 很 多 种 ， 但 是 常用 的 是 MySQL 数据 库 。 

将 PHP 和 MySQL 合理 且 有 效 地 结合 在 一 起 可 以 开发 精致 的 动态 交互 网 站 。MySQL 是 一 种 小 
型 的 ,紧密 的 数据 服务 器 ,支持 标准 SQL, 它 在 UNIX 和 Windows 环境 下 都 能 够 使 用 ,PHP 和 MySQL 
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都 是 免费 的 开放 源码 ,它们 的 结合 可 以 在 Windows 中 发 展 ， 也 可 以 在 UNIX 中 服务 。PHP 也 支持 其 
他 一 些 数据 库 ， 如 PostgreSQL 。 

在 PHP 中 有 一 个 专门 的 函数 库 ， 这 些 函数 提供 了 到 MySQL 数据 库 管 理 系统 的 接口 。 使 用 它们 
可 以 访问 和 修改 驻 留 在 MySQL 服务 器 中 的 数据 库 。 


1.2 ”安装 支持 软件 


上 面 的 章节 介绍 了 PHP 的 起 源 、 发 展 、 特 点 等 ， 这 些 都 能 帮助 我 们 简单 认识 PHP 技术 。 现 在 
自己 动手 配置 PHP 运行 环境 ，PHP 的 运行 环境 需要 两 种 软件 的 支持 ， 一 个 是 PHP 脚本 运行 的 服务 
器 Apache Web, 一 个 是 PHP 页 面 运行 时 需要 分 析 PHP 代码 的 软件 PHP 链接 库 。 本 节 将 学 习 安装 和 
配置 PHP 的 知识 ， 并 在 这 个 过 程 中 学 习 安装 Apache Web 服务 器 。 


1.2.1 下 载 Apache 和 PHP 


在 开始 安装 之 前 ， 需 要 获取 这 两 种 软件 的 安装 文件 。 
1. 下 载 Apache 
Apache 是 世界 排名 第 一 的 Web 服务 器 ， 根 据 netcraft (www.netcraft.co.uk〉 所 作 的 调查 ， 世 界 
上 50% 以 上 的 Web 服务 器 使 用 Apache。 最 早 的 Apache (0.6.2 版 本 ) 由 Apache group 公布 发 行 ， 
Apache group 是 一 个 完全 通过 Internet 进行 运作 的 非 盈 利 机 构 ， 由 它 来 决定 Apache Web 服务 器 的 标 
准 发 行 版 本 中 应 该 包含 哪些 内 容 ， 准 许 任何 人 修改 隐 错 ， 提 供 新 的 特征 和 将 它 移植 到 新 的 平台 上 ， 
以 及 其 他 的 工作 。 当 新 的 代码 被 提交 给 Apache group 时 ， 该 团体 审核 它 的 具体 内 容 ， 进 行 测试 ， 如 
果 认 为 满意 ， 该 代码 就 会 被 集成 到 Apache 的 主要 发 行 版 本 中 。 
Apache 的 特性 如 下 所 示 : 
几乎 可 以 运行 在 所 有 的 计算 机 平台 上 。 
支持 http/1.1 通信 协议 。 
简单 而 且 强 有 力 的 基于 文件 的 配置 (httpd.conf )。 
支持 通用 网 关 接 口 (CGI)。 
支持 虚拟 主机 。 
支持 HTTP 认证 。 
集成 Perl。 
集成 的 代理 服务 器 。 
可 以 通过 Web 浏览 器 监视 服务 器 的 状态 ， 可 以 自 定义 日 志 。 
支持 服务 器 端 包含 命令 ( SSI)。 
支持 安全 Socket 层 (SSL )。 
具有 用 户 会 话 过 程 的 跟踪 能 力 。 
支持 FastCGI。 
支持 Java Servlets。 
Apache 的 缺点 是 没有 为 管理 员 提供 图 形 用 户 接口 《GUI)， 但 新 的 Apache 版 本 已 经 有 了 GUI 


的 支持 。 

Apache 的 版 本 更 新 速度 非常 快 ， 并 且 在 多 种 操作 系统 上 都 可 以 安装 Apache。 基 于 上 面 的 原因 ， 
下 载 Apache 服务 器 时 ， 应 该 从 Apache 的 官方 网 站 http://www.apache.org 下 载 ， 打 开 正 浏览 器 ， 在 
地 址 栏 中 输入 该 地 址 ， 单 击 【 转 到 】 按 钮 ， 会 显示 如 图 1-1 所 示 的 窗口 。 
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图 1-1 Apache 网 站 首页 
单 击 Http Server 超级 链接 ， 会 显示 如 图 1-2 所 示 的 窗口 。 
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图 1-2 Apache 2.2.4 显示 页 面 
单 击 Download 超级 链接 ， 会 显示 如 图 1-3 所 示 的 窗口 。 
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在 该 页 面 中 可 以 看 到 Apache 2.2.4 版 本 有 两 种 形式 ， 一 种 以 源 代 码 的 形式 存在 ， 一 种 以 二 进 制 
代码 的 形式 存在 。 如 果 使 用 的 是 Linux 操作 系统 ， 这 里 选择 下 载 源 代码 形式 ， 这 样 会 在 Linux 系统 
重新 生成 可 执行 文件 ， 使 用 起 来 非常 方便 。 本 书 使 用 的 是 Windows 操作 系统 ， 这 里 有 两 种 方式 可 供 
选择 ， 一 种 是 源 代码 形式 ， 一 种 是 二 进 制 文件 。 这 里 选择 二 进 制 文件 可 执行 程序 下 载 ， 单 击 Win32 
Binary (MSI Installer) 超级 链接 ， 下 载 源 文件 。 

2. 下 载 PHP 

PHP 开发 工具 包 是 PHP 程序 在 运行 时 ， 需 要 加 载 的 一 个 主要 软件 包 , 该 软件 包 主要 是 解释 执行 
PHP 页 面 的 脚本 程序 ， 如 解释 PHP 页 面 的 函数 。 虽 然 现在 大 多 数 Linux 发 行 包 都 捆绑 了 PHP， 但 还 
是 应 当 从 PHP 网 站 下 载 最 新 的 稳定 版 本 。 

为 减少 下 载 时 间 ， 可 从 位 于 50 多 个 国家 和 地 区 的 100 多 个 官方 镜像 中 选择 下 载 ， 这 个 镜像 列 
表 位 于 网 站 http://www.php.net/mirrors.php。 打 开 正 浏览 器 ， 在 地 址 栏 中 输入 该 地 址 ， 会 显示 如 图 
1-4 所 示 的 窗口 。 
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图 1-4 PHP 下 载 镜像 
单 击 cn.php.net 超级 链接 ， 会 显示 如 图 1-5 所 示 的 窗口 。 
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图 1-5 PHP 下 载 网 站 


单 击 导航 栏 上 的 downloads 超级 链接 ， 会 显示 如 图 1-6 所 示 的 窗口 。 
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图 1-6 PHP 下 载 选项 页 面 


在 该 窗口 中 ， 可 以 观察 到 PHP 的 发 行 包 有 三 种 形式 , 一 种 是 源 代 码 形式 的 ， 该 类 型 的 代码 的 服 
务 器 平台 是 UNIX， 或 者 计划 在 Windows 平台 上 编译 源 代码 ， 可 以 选择 这 种 发 行 包 格式 ， 一 种 是 
Windows 下 的 zip 包 ， 这 种 二 进 制 包括 CGI 和 各 种 服务 器 模块 ， 如 果 计 划 在 Windows 下 结合 使 用 
PHP 和 Apache， 应 当下 载 这 个 包 ; 最 后 一 种 是 Windows 下 的 安装 程序 ， 它 只 包括 CGI 二 进 制 包 ， 
为 安装 和 配置 PHP 提供 了 一 个 方便 的 Windows 安装 程序 界面 ， 支 持 TS、PWS 和 Xitami 服务 器 的 
自动 配置 ， 整 个 过 程 不 需要 手动 操作 。 虽 然 也 可 以 与 Apache 一 起 使 用 ， 但 不 推荐 这 种 做 法 。 如 果 
要 与 Apache 一 起 使 用 , 可 选择 Windows 下 的 zip 包 版 本 。 这 里 选择 zip 包 , 单 击 PHP 5.2.3 zip package 
超级 链接 ， 会 显示 如 图 1-7 所 示 的 窗口 。 


文件 @)。 生 入 尼 ) 坦 看 [0 收 帘 仙 | 工具 GD) 条 对 td) 
从 hy /jen phy nevVcevphi-5.2 :finac_ziPy froa/ayatrror 


Choose mirror site for download 


You have chosen to downloed the following fle: 


Php-5.2.3-Win32.2ip 
9.857,499 Dyies 


Please chose the mirror closest to You from Which tp download the fle, The ourrent mrror ts highhighted In yellow, mirror 
Sltes detected to be out of date or dysfunctional are not listed for your convenience. 


We have automatically detected the folowing mirror to be dose to you If you Use & mrirror cose to you for dowrloads and 
Your usual daily work, you will get bettsr response times, 


国 chna 


» cn.nhp net YeryCD Network 


Here is the ist of all the pther sites In case our detection did something wrong, or the local mirror sites are busy. 
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图 1-7 PHP 下 载 页 面 
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单 击 cn php net 超级 链接 ， 会 自动 执行 下 载 程序 。 至 此 ， 所 需要 的 软件 已 经 下 载 完 毕 。 
1.2.2 ”安装 Apache 和 PHP 


构建 PHP 程序 的 运行 环境 时 , 由 于 操作 系统 的 不 同 ,会 有 不 同 的 安装 方式 .本 节 介绍 在 Windows 
下 构建 PHP 运行 环境 。 

1. 安装 Apache 

首先 安装 Apache 服务 器 ， 双 击 下 载 的 软件 包 apache_ 2.2.4-win32-x86-no_ssl.msi， 启 动 Apache 
安装 程序 ， 会 显示 如 图 1-8 所 示 的 对 话 框 ， 该 对 话 框 为 Apache 服务 器 的 欢迎 使 用 界面 。 单 击 Next 
按钮 ， 会 显示 如 图 1-9 所 示 的 对 话 框 。 


Welcome to the Installation wizard for Lim dent 
Apache HTTP Server 2.2.4 Fesse redte folowng errse ayeemen corcidh, 


Ie maioren ye wnatal Apacre HTTP Sever ZZ. en Apache License 
your onpurer, To cortinus, Version 20, January 2004 
mpsywwwanacheorancenses/ 
TERMS AND CONDMTIONS FOR LISE, REPRODUCTION, AND DISTRIBUTIDN 
.Dorntione, 


Wa the preg mie proteand by ceprrght lew md hems shal moan he erms and condtons for vss, roproducton and 
Wormatonal ye ss cofined by Sartione 1 througn 9 ofthis dorumert 


Or 和 net vecore the ternein the lceree greemert 


Cw J Nex> 


图 1-8 Apache 安装 欢迎 界面 图 1-9 ”Apache 接受 协议 


单 击 图 1-9 所 示 的 复 选 框 , 该 对 话 框 主要 是 安装 的 许可 协议 , 这 里 同意 该 协议 , 选中 “I accept the 
terms in the license agreement” 单 选 按钮 ， 然 后 单 击 Next 按钮 ， 会 显示 如 图 1-10 所 示 的 对 话 框 ， 其 
中 主要 是 对 Apache 服务 器 的 介绍 。 单 击 Next 按钮 ， 会 显示 如 图 1-11 所 示 的 对 话 框 。 


evakConan eg somoret.com) 
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TREE 
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Ny dere 


ednev dors tnal Adcess (e 9 webmastergsomenes, om): 
soh .con 


nel Apeche HTTP Server 2,2 progr ons and shortouts For; 


Ofor Busers, on Fort 80, &s 3 5ervce ~ Reccnimended, 
© en ter the Cament Uset, on Port B000, when started Menualy. 


Cs J Cems Er TE | 


1-10 Apache 介绍 图 1-11 Apache 选项 输入 


第 一 个 文本 框 表示 计算 机 的 网 络 域 , 如 果 默 认 ， 就 表示 是 本 机 P 地 址 ; 第 二 个 文本 框 表 示 服 务 
器 的 名 称 ， 这 里 可 以 设 定 ， 如 果 默 认 ， 就 表示 为 localhost; 第 三 个 文本 框 是 系统 管理 员 的 电子 邮件 
地 址 ， 这 里 可 以 输入 自己 的 邮件 地 址 ， 以 后 再 进行 修改 。 界 面 下 方 提示 是 为 所 有 用 户 提供 Apache 


服务 ， 还 是 仅 为 当前 用 户 提 供 服务 ， 这 里 选中 only for the Current User，on Port 8080，when started 
Manually 单 选 按钮 。 单 击 Next 按钮 ， 会 显示 如 图 1-12 所 示 的 对 话 框 ， 在 该 对 话 框 中 可 以 选择 安装 
类 型 ， 有 典型 安装 和 定制 安装 。 这 里 选择 典型 安装 。 单 击 Next 按钮 , 会 显示 如 图 1-13 所 示 的 对 话 框 。 


彰 tpache HTIP Server 2. 2 - Installation 
Setup Type 
Chacse the setup type thar best suts your needs, 


Destination Folder 
Cick ranoe torstal Da drferent olcer 


Hoaso soloct 3 setup typo, 


EPE 
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一 [臣下 和 
Erogenmiesypede Scitware FoundetioMApadhe2 2 


Ogustom 
Chocse which progran lealures you want nstaled and whrere they 
wil be netaled Rocommnended For hanced ucere 


图 1-12 安装 类 型 


在 该 对 话 框 中 可 以 自由 设 定 Apache 的 安装 目录 ， 这 里 选择 C:\Web\apache 目录 下 。 单 击 Next 
按钮 , 会 显示 如 图 1-14 所 示 的 对 话 框 , 单 击 Install 按钮 , 开始 安装 Apache 服务 器 , 会 显示 如 图 1-15 
所 示 的 对 话 框 。 


图 1-13 安装 路 径 


pal the Program 
The weard teady to beon natalatcn， 


chekInetal to bogn the netallaton, 
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Dey take Several ines, 
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EA 


图 1-14 准备 安装 


图 1-15 开始 安装 


安装 完成 之 后 ， 会 显示 如 图 1-16 所 示 的 提示 对 话 框 ， 
提示 成 功 安装 了 Apache 服务 器 。 


这 时 会 在 右 下 角 的 状态 栏 中 出 现 而 图 标 ,表示 Apache 忆 


已 经 成 功 安装 了 ， 这 时 ， 单 击 该 图 标 ， 会 出 现 Eee 图 
标 ， 单 击 该 图 标 会 出 现 _* | 图标， 在 此 处 单 击 Start 图 标 
就 可 以 启动 Apache 了 。 成 功 启动 后 ， 会 出 现 轿 图 标 。 还 
可 以 使 用 另外 一 种 方式 启动 Apache 服务 器 ， 打 开 命 令 提 
示 符 窗口 , 进入 C:\Web\apache\bin 安装 目录 下 , 在 窗口 中 
输入 httpd 命令 ， 会 显示 如 图 1-17 所 示 的 窗口 。 

检验 Apache 是 否 安装 成 功 , 打开 正 浏览 器 ,在 地 址 


Installation Wizard Completed 


Ihe irstalanon ward bas suceessiuly nstaled Roache HTTP 
Sorver 2.2.4, Cd Fhich to srt the viard, 


1-16 ”安装 成 功 
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栏 中 输入 http://localhost:8080， 单 击 【 转 到 】 按 钮 ， 会 显示 如 图 1-18 所 示 的 窗口 。 


图 1-17 使 用 命令 启动 Apache 


2. 安装 及 配置 PHP 

当 出 现 图 1-18 中 的 信息 时 ,表示 Apache 已 经 安装 成 功 。 现 
在 可 以 安装 并 配置 PHP。 其 安装 过 程 如 下 : 

(1) 解压 PHP 包 ， 将 解压 后 的 内 容 全 部 复制 到 安装 的 目录 
下 ， 这 里 复制 到 C:\Web\php 目录 下 ， 这 里 不 要 使 用 带 有 空格 的 
路 径 。 

(2) 配置 Apache 运行 时 需要 加 载 的 php5ts.dll 文件 ,最 简单 
的 方法 是 将 PHP 的 安装 路 径 追 加 到 Windows 系统 中 path 路 径 的 
下 面 。 这 里 的 路 径 是 【我 的 电脑 】|【 属 性 】|【 高 级 】| 【环境 变 


ho iC: Peer mn 了 les 


量 】， 在 【环境 变量 】 对 话 框 中 ， 找 到 path 路 径 ， 单 击 【 编 辑 】 i 


按钮 ， 在 【编辑 系统 变量 】 对 话 框 中 ， 将 C:\Web\php 追加 到 路 | 


径 中 ， 如 图 1-19 所 示 。 a 
(3) 打开 C:\Web\apache\conf 目录 ， 找 到 httpd.conf 文件 ， 
打开 该 文件 ， 增 加 下 面 三 行内 容 : 图 1-19 修改 path 路 径 


LoadModule php5 module "C:\Web\php\\php5apache2 2.d11" 
AddType application/x-httpd-php .php 
PHPIniDir "C:\Web\php" 


在 该 段 代 码 中 ,第 一 行 表示 要 加 载 的 模块 在 哪个 位 置 存储 ; 第 二 行 表示 将 一 个 MIME 类 型 绑 定 
到 某 个 或 某 些 扩展 名 。.php 只 是 一 种 扩展 名 ， 这 里 可 以 设 定 为 html、.php2 等 ; 第 三 行 表 示 PHP 所 
在 的 初始 化 路 径 。 


与 ”这 里 可 以 定义 多 个 扩展 名 ，.php 只 是 一 种 建议 ， 还 可 以 使 用 .html、.php5 等 。 有 些 用 户 喜欢 使 用 html 
提示 | 扩展 名 ， 这 样 会 导致 每 次 请 求 HTML 文件 时 都 会 把 文件 交 由 PHP 解析 ， 降 低 了 PHP 的 性 能 。 


(4) 将 C:\Web\php 目录 下 的 php.ini-dist 文件 重 命名 为 php.ini。 在 php.ini 中 包含 了 很 多 负责 调 
整 PHP 行为 的 指令 。 至 此 PHP 运行 的 Windows 平台 已 经 安装 完成 了 。 


1.2.3 测试 PHP 环境 


如 果 要 验证 上 面 的 PHP 安装 是 否 成 功 ， 最 好 的 方法 就 是 执行 一 个 带 有 PHP 脚本 的 程序 。 打 开 
记事 本 ， 输 入 下 列 代码 : 


<?php 
phpinfo () 7 
多 


将 该 文件 保存 在 C:\Web\apache\htdocs， 文 件 名 为 info php。 打 开 正 浏览 器 ， 在 地 址 栏 中 输入 
http://localhost:8080/info.php， 单 击 【 转 到 】 按 钮 ， 会 显示 如 图 1-20 所 示 的 窗口 。 


文件 中 六 辑 人 E) 查看 Q) 收 巷 和 工具 加 ) 
地 址 中 ) 虱 http;//1ocalhest;8080/info_ php 


图 1-20 PHP 运行 测试 


如 果 出 现 该 窗口 ， 表 示 Apache 和 PHP 的 配置 一 切 正常 。Phpinfo0 函 数 会 提供 与 PHP 安装 有 关 
的 一 组 有 用 的 信息 。 


1.2.4 Windows 下 扩展 PHP 


PHP 的 Windows 安装 共有 45 个 扩展 包 , 都 位 于 INSTALL _DIR\ext 目录 下 。 通 过 这 些 包 的 使 用 ， 
可 以 增加 新 功能 。 但 是 ， 要 真正 使 用 这 些 扩展 包 ， 还 需要 进行 相应 的 修改 ， 如 取消 php.ini 文件 中 的 
相应 注释 。 例 如 ， 如 果 希 望 启用 PHP 的 IMAP 扩展 ， 需 要 在 php.ini 文件 中 进行 两 个 很 小 的 调整 ， 
其 步骤 如 下 所 示 : 

(1) 打开 位 于 Windows 目录 下 的 php.ini 文件 。 要 确定 php.ini 文件 在 哪 一 个 目录 下 。 需 要 找到 
extension_dir 指令 ， 将 其 赋 为 C:\php5\ext。 如 果 在 其 他 目录 中 安装 PHP， 则 要 进行 相应 的 修改 。 

(2) 找到 代码 行 : ;extension=php_imap.dll。 删 除 前 面 的 分 号 ， 取 消 这 一 行 的 注释 。 保 存 并 关闭 
文件 。 

(3) 重新 启动 Apache， 就 可 以 在 PHP 中 使 用 这 个 扩展 了 。 注 意 ， 有 些 扩展 在 正常 使 用 前 还 需 
要 进行 更 多 的 修改 。 


1.2.5 ”常见 错误 


第 一 次 在 线 访问 PHP 页 面 时 , 经 常会 遇 到 一 些 初 级 的 问题 。 这 一 节 将 讨论 其 中 最 常见 的 一 些 问 
题 ， 其 详细 信息 如 表 1-3 所 示 。 


全 有 妓 dHd 册 一 浊 


表 1-3 常见 错误 


错误 提示 


错误 原因 


解决 办 法 


只 能 通过 命令 启动 ， 不 能 
随 Windows XP 系统 一 起 


因为 Windows XP 不 是 专业 的 服务 器 操作 系 
统 (虽然 它 也 有 “服务 ”这 一 项 功能 )， 有 了 时 


要 做 以 下 工作 : 通过 Windows XP 的 
控制 台 , 进入 Apache 安装 目录 的 bin 


启动 可 能 会 出 现 安装 后 没有 服务 器 可 以 启动 。 子 目录 下 ， 使 用 如 下 命令 将 Apache 
此 ,安装 时 虽然 选择 了 8080 端口 作为 Apache | 安装 为 Windows NT 服务 : httpd-k 
端口 ,但 是 Apache 并 没有 被 安装 到 【控制 面 | install。 此 时 ， 可 以 查看 【服务 】 中 
板 】|【 管 理工 具 】|【 服 务 】 中 有 了 Apache 2.2 的 服务 项 

配置 文件 修改 后 不 生效 ”| 修改 Apache 的 配置 文件 后 , 在 重新 启动 前 并 | 在 向 这 个 配置 文件 增加 了 必要 的 


不 会 立即 生效 


PHP 内 容 后 ， 要 确保 重启 Apache 


Apache 文件 修改 , 无 法 启 
动 服务 器 


修改 Apache 的 配置 文件 时 ,可 能 会 引入 非法 
字符 ， 导 致 Apache 无 法 重启 


如 果 Apache 无 法 启动 ， 可 查看 一 下 
所 进行 的 修改 


不 识别 PHP 文件 扩展 名 


在 浏览 器 中 输出 PHP 代码 


默认 情况 下 ， 不 能 启动 
index.php 文件 
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http://httpd.apache.org， 单 击 【 转 到 】 按 钮 ， 


veleone! - The Apache HTIP Sorver Pr 


验证 文件 的 扩展 名 确实 是 httpd.conf 文件 中 
所 指定 的 PHP 特定 扩展 名 ， 如 果 只 定义 
了 .php 作为 可 识别 的 扩展 名 ， 就 不 要 在 .html 
中 嵌入 PHP 代码 

在 PHP 文件 中 ， 没 有 界定 PHP 代码 


创建 名 为 index.php 的 文件 后 , 力图 作为 默认 
目录 索引 来 调用 时 ， 将 不 会 成 功 。 注 意 ， 默 
认 情 况 下 Apache 只 能 识别 index.html 


1.2.6 ”查看 并 下 载 文档 


修改 httpd.conf 文 件 


修改 httpd.conf 文件 (文件 界定 符 在 
第 二 章 中 将 会 详细 介绍 ) 

向 Apache 的 DirectoryIndex 指令 增加 
index.php 


Apache 和 PHP 软件 都 提供 了 相应 的 说 明文 档 ， 用 来 介绍 这 两 种 技术 的 使 用 和 配置 。 通 过 说 明 
文档 可 以 非常 容易 地 掌握 Apache 和 PHP 的 使 用 。 这 些 文档 分 别 在 http://httpd.apache.org 和 
http://www.php.net 网 站 上 。 在 这 里 可 以 查看 最 新 的 版 本 ， 并 且 可 以 下 载 到 本 地 计算 机 上 学 习 。 

1. 查看 Apache 手册 

每 一 个 Apache 版 本 出 现 之 后 ， 都 有 相应 的 说 明文 档 ， 用 来 帮助 用 户 更 好 、 更 快 地 掌握 该 工具 。 
Apache 使 用 手册 可 以 在 线 阅 读 或 者 下 载 到 本 地 计算 机 阅读 。 打 开 正 浏览 器 ， 在 地 址 栏 中 输入 
会 显示 如 图 1-21 所 示 的 窗口 。 


14 


图 1-21 Apache 网 站 首页 


单 击 Version 2.2 超级 链接 ， 会 显示 如 图 1-22 所 示 的 窗口 。 


| 
文件 中 。 纲 术 他 查理 YY 收藏 入) 工具 ID 大 肋 0 
把 引 四 加 wip /tps wee wore 


Apache HTTP Server Version 2.2 Documentation 


Palahe Languages de en es 


Release Notes " How-To /Tutorials 


aumorzancn ang 


ss (pthc_ pam 


i 
EARLY Platform Soecific Netes 
CE 


图 1-22 Apache 使 用 手册 


在 该 窗口 中 ， 可 以 发 现 该 文档 是 以 HTML 格式 存在 的 ， 文 档 的 语言 类 型 有 英语 、 法 语 、 德 语 、 
日 语 、 韩 语 、 俄 语 共 6 种 类 型 。 该 文档 的 目录 位 于 dosc 的 下 面 。 如 果 对 Apache 的 使 用 有 什么 疑难 
问题 ， 可 以 直接 在 此 处 查询 。 

2. 查看 PHP 手册 

PHP 文档 是 使 用 PHP 的 说 明文 档 。 该 文档 可 以 下 载 也 可 以 在 线 阅 读 。PHP 文档 支持 24 种 语言 ， 
并 包括 各 种 格式 ， 如 单个 的 HTML 页 面 、 多 个 HTML 页 面 、Microsoft HTML 帮助 格式 ， 以 及 扩展 
的 HTML 格式 。 这些 版 本 都 是 由 基于 Docbook 的 主 文件 生成 的 。 如 果 需 要 转换 为 其 他 的 格式 ， 可 以 
在 PHP 项 目的 CVS 服务 器 获得 这 些 文件 。PHP 文档 位 于 安装 根 目录 的 manual 目录 下 。 

打开 正 浏览 器 ， 在 地 址 栏 中 输入 http://www.php.net， 单 击 【 转 到 】 按 钮 ， 会 显示 如 图 1-23 所 
示 的 窗口 。 
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单 击 documentation 超级 链接 ， 会 显示 如 图 1-24 所 示 的 窗口 。 


加 PHP: Documentation - Wicrosoft Internet Ezplorer 


文件 四 凡 辑 到 豆 看 如 收 京 册 工具 0D) 帮 动 加 


地 址 加 | 图 http: /frm php_net/docs. php 
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licensing questions see the 
separate Fat 
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，Documentation 


The PHP Manual is available online in a selection oflanguages, Please pick a language 
from the list balow. 


You can learn how ta integrate our online manual with various tools, induding your 
Web browser, on Our quick reference tips page. You can also get more information 
about php.net URL shortcuts by visiting our URL howto page. 


| Note, that many languages are just under translation, and the untranslated parts are 


stll in English, Also some transjated parts might be outdated, The translation teams 
are open to contributions. 


ormats Destinations 
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图 1-24 语言 类 型 选择 


单 击 Chinese (Simplified) 超级 链接 ， 会 显示 中 文 的 PHP 使 用 说 明文 档 ， 如 图 1-25 所 示 。 


加 PHP: PHP 手册 - anual 


Ficrosoft Internet Explorer 加 EE 
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1-25 ”PHP 使 用 文档 


该 窗口 中 ， 列 出 了 PHP 的 相关 设 定 和 配置 说 明 ， 以 及 PHP 的 基本 语法 等 。 


1.3 ”配置 环境 


进行 完 前 面 的 操作 后 ,就 可 以 使 用 Apache 和 PHP 了, 但 这 时 运行 的 PHP 页 面 都 是 在 环境 默认 
的 情况 下 运行 的 。 如 果 要 在 PHP 页 面 中 增加 一 些 新 的 功能 ， 如 限制 使 用 文件 下 载 功 能 ， 就 需要 对 
PHP 的 配置 进行 修改 。 大 多 数 的 修改 基本 上 都 是 在 Apache 的 httpd.conf 和 PHP 的 php.ini 文件 中 进 
行 的 ， 这 两 个 文件 包含 了 大 量 的 配置 指令 ，Apache 和 PHP 的 行为 分 别 由 其 相应 文件 中 的 配置 指令 
控制 。 配 置 指令 决定 了 PHP 在 运行 时 出 现 的 形式 ， 如 可 以 定义 PHP 的 编码 形式 。 本 节 将 详细 介绍 
PHP 中 最 常用 的 配置 指令 ， 并 介绍 这 些 指令 的 作用 、 作 用 域 和 默认 值 。 


1.3.1 管理 PHP 的 配置 指令 


配置 指令 的 使 用 是 为 了 使 PHP 功能 更 加 强大 , PHP 运行 更 加 方便 。 对 配置 指令 的 熟练 使 用 程度 
标志 着 对 PHP 掌握 的 多 少 。 在 学 习 各 个 配置 指令 之 前 ， 需 要 学 习 如 何 设置 和 管理 这 些 相应 的 配置 指 
令 。 在 PHP 技术 中 有 下 面 几 种 方法 可 以 设置 配置 指令 的 值 , 分 别 为 修改 php.ini、 httpd.conf、 .htaccess 
文件 ， 也 可 以 直接 通过 PHP 脚本 处 理 。 
1. php.ini 
PHP 有 两 个 配置 模板 : php.ini-dist 和 php.ini-recommended。 建 议 使 用 后 者 ， 因 为 其 中 的 许多 参 
数 都 已 经 设置 为 推荐 值 。 如 果 采 纳 这 个 建议 ， 在 保证 安装 安全 以 及 调整 安装 时 ， 就 能 节省 大 量 的 时 
间 和 精力 ， 因 为 这 个 文件 中 有 大 约 240 个 不 同 的 配置 参数 。 虽 然 默 认 值 有 助 于 快速 地 部 署 PHP， 但 
是 如 果 还 想 对 PHP 的 行为 做 另外 的 调整 , 就 有 必要 对 这 个 文件 有 所 了 解 , 学 习 其 中 的 许多 配置 参数 。 
与 Apache 的 httpd.conf 文件 或 MySQL 的 my.cnf (Windows 下 是 my.ini) 类 似 ，php.ini 文件 是 
PHP 的 全 局 配置 文件 。 这 个 文件 处 理 了 PHP 在 12 个 不 同方 面 的 行为 ， 分 别 如 下 所 示 : 
。 语言 选项 。 
安全 模式 。 
语法 突出 显示 。 
杂项 。 
资源 限制 。 
错误 处 理 和 日 志 。 
数据 处 理 。 
路 径 和 目录 。 
文件 上 传 。 
Fopen 包装 器 。 
动态 扩展 。 
。 模块 设置 。 
php.ini 文件 是 一 个 纯 文 本 文件 ， 只 包含 注释 和 “参数 = 值 ” 赋 值 对 。 下 面 是 php.ini 文件 中 的 一 
个 示例 片段 : 


泛 
?Safe Mode 
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; 
Safe mode=Off 


以 分 号 开头 的 行 是 注释 ， 这 里 safe_ mode 参数 的 值 赋 为 Off。 

如 果 很 清楚 某 个 配置 参数 的 作用 ， 可 以 将 其 注释 删除 ， 使 文件 的 内 容 简化 ， 从 而 减少 以 后 的 编 
辑 时 间 。 

修改 将 何 时 生效 ， 这 取决 于 安装 PHP 的 方式 。 如 果 作 为 CGI 二 进 制 包 安 装 PHP， 那 么 每 次 调 
用 PHP 时 都 会 重新 读 取 php.ini 文件 ， 因 此 修改 将 立即 生效 。 如 果 作 为 Apache 模块 安装 PHP， 则 只 
会 在 Apache 守护 进程 第 一 次 启动 时 读 取 php.ini。 因 此 ， 如 果 按 照 后 一 种 方式 安装 PHP， 就 必须 重 
启 Apache， 这 样 修改 才 会 生效 。 

2. Apache 的 httpd.conf 和 .htaccess 文件 

当 PHP 作为 Apache 模块 运行 时 ， 可 以 通过 httpd.conf 或 .htaccess 文件 修改 指令 。 为 此 ， 可 以 在 
“名 = 值 ”对 前 面 加 上 以 下 某 个 关键 字 作 为 前 级 ， 其 详细 信息 如 表 1-4 所 示 。 


表 1-4 指令 信息 表 
说 了 明 


指令 格式 
php_value 
php_flag 
php_admin_ value 


设置 指定 指令 的 值 

设置 指定 布尔 指令 的 值 

设置 指定 指令 的 值 。 它 与 php_value 不 同 ， 不 能 用 在 .htaccess 文件 中 ， 也 不 能 在 虚拟 主机 
或 .htaccess 中 被 覆盖 


设置 指定 布尔 指令 的 值 。 它 与 php_flag 不 同 ， 不 能 用 在 .htaccess 文件 中 ， 也 不 能 在 虚拟 主机 
或 .htaccess 中 被 覆盖 


3. 在 执行 脚本 中 
第 三 种 处 理 PHP 配置 变量 的 方式 是 通过 ini_set0 方 法 完成 ， 这 也 是 最 本 地 化 (localized) 的 方 
式 。 例 如 ， 假 设 要 修改 PHP 中 给 定 脚本 的 最 大 执行 时 间 ， 只 需 在 脚本 最 上 面 加 入 如 下 命令 : 


php_admin flag 


ini set("max execution time","60"); 


4. 配置 指令 作用 域 
并 不 是 在 任何 地 方 都 能 修改 配置 指令 ， 原 因 有 很 多 ， 大 多 与 安全 有 关 。 每 个 指令 都 有 自己 的 作 
用 域 ， 指 令 只 能 在 其 作用 域 中 修改 。 共 有 4 个 作用 域 ， 其 详细 信息 如 表 1-5 所 示 。 


表 1-5 指令 作用 域 
指令 名 称 作用 域 说 明 
PHP _ INI PERDIR 指令 可 以 在 php.ini、httpd.conf 或 htaccess 文件 中 修改 
PHP INI SYSTEM 指令 可 以 在 php.ini 和 httpd.conf 文件 中 修改 
PHP INI USER 指令 可 以 在 用 户 脚 本 中 修改 
PHP INI ALL 指令 可 以 在 任何 地 方 修改 


1.3.2 ”PHP 的 配置 指令 


上 面 的 小 节 中 介绍 了 怎样 配置 和 管理 这 些 配置 指令 ， 本 节 将 对 常用 的 配置 指令 进行 说 明 。 


1. 语言 选项 
该 部 分 指令 主要 用 于 确定 语言 最 基本 的 一 些 行为 。 其 配置 的 详细 信息 如 表 1-6 所 示 。 


表 1-6 语言 选项 指令 


指令 作 用 域 说 了 明 
engine(On, Off) PHP INI ALL 默认 值 为 On 确定 PHP 引擎 是 否 可 用 
zend.zel_compatibility_ mode PHP_INI_ALL 默认 值 为 On PHP 5.0 和 PHP 4.0 之 间 存 在 不 兼容 的 特性 。 
(on. of 如 果 启 动 该 指令 ,可 以 在 PHP 5.0 中 运行 PHP 
4.0 的 程序 


short_open_tag(On, Off) PHP INI ALL 默认 值 为 On 可 以 在 PHP 文件 中 使 用 段 标记 <??> 界 定 
PHP 代码。 如 果 要 和 XML 结合 使 用 PHP， 
可 以 禁用 此 选项 以 便于 嵌入 使 用 <?xml ?> 


asp_tags(On, Of PHP_INI_ALL 默认 值 为 Of 用 来 设置 是 否 支持 ASP 风格 的 界定 符 
precision(integer) PHP INI ALL 默认 值 为 12 设置 浮 点 数 中 显示 的 有 效 数 字 个 数 
y2k_compliance(On, Of PHP_INI_ALL 默认 值 为 Of 禁用 y2k_compliance 参数 


output_buffering(On, Of 用 来 设 定 是 否 使 用 缓冲 ， 启 用 output_ 
buffering 指令 将 打开 缓冲 

output_ handler 用 来 设置 把 输出 返回 给 请 求 用 户 之 前 ， 会 将 
输出 传递 给 一 个 函数 

allow_call time pass_ Teference 函数 有 两 种 传 值 方式 ， 按 值 和 引用 。 此 指令 

(On. Of 决定 可 以 在 函数 中 指定 每 个 参数 在 函数 调 
用 时 如 何 传递 参数 

serialize_precision(integer) 确定 在 串 行 化 双 精度 和 单 精度 浮 点 数 时 小 
数 点 后 存储 的 位 数 

implicit_fhush 启用 该 命令 后 ， 每 次 调用 print 或 echo， 以 
及 每 个 颈 入 的 HTML 块 来 完成 后 , 将 会 自动 
刷新 或 清除 其 内 容 的 输出 缓冲 区 


2. 资源 限制 

PHP 5.0 在 处 理 资源 功能 方面 有 长 足 的 进步 ， 但 在 执行 程序 时 还 要 确保 PHP 脚本 不 会 由 于 程序 
员 或 用 户 的 动作 而 独占 服务 器 资源 。 有 三 个 方面 可 能 会 过 度 耗费 资源 ， 分 别 为 脚本 执行 时 间 、 脚 本 
输入 处 理 时 间 和 内 存 。 资 源 消耗 可 以 使 用 下 面 三 个 指令 来 控制 ， 其 详细 信息 如 表 1-7 所 示 。 


表 1-7 资源 限制 指令 


设置 PHP 脚本 执行 时 间 的 上 限 ， 以 秒 为 单位 。 如 果 设 置 为 0， 
将 取消 最 大 限制 

设置 PHP 脚本 解析 请 求 数据 所 用 的 时 间 ， 以 秒 为 单位 。 在 文件 
上 传 时 ， 该 参数 特别 有 用 

设 定 了 一 个 脚本 所 能 够 申请 到 的 最 大 内 存 字 节 数 。 这 有 助 于 防 
止 写 得 不 好 的 脚本 消耗 完 服务 器 上 的 可 用 内 存 。 要 使 用 此 指令 
必须 在 编译 时 激活 


max_execution time(integer) | PHP_ INI ALL 
默认 值 为 30 

PHP INI ALL 
默认 值 为 60 
PHP INI ALL 
默认 值 为 SMB 


max_input_time(integer) 


memory_limit(integer)M 
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3. 安全 模式 

在 多 用 户 环境 中 部 署 PHP 时 ， 可 能 要 限制 PHP 的 功能 。 因 为 在 多 用 户 环境 中 ， 如 果 为 每 个 用 
户 都 提供 PHP 的 所 有 功能 ， 就 会 暴露 服务 器 的 漏洞 ， 还 可 能 破坏 服务 器 的 资源 和 文件 ， 故 PHP 应 
采用 一 种 受 限 模式 或 安全 模式 运行 。 但 启用 安全 模式 会 有 很 多 影响 ， 包 括 自动 禁用 很 多 功能 和 可 能 
不 安全 的 各 种 特性 。 可 以 使 用 函数 进行 限制 ， 也 可 以 通过 在 PHP 中 配置 指令 进行 限制 。 其 详细 信息 
如 表 1-8 所 示 。 


表 1-8 安全 模式 指令 


指令 作 用 域 说 有明 
safe_ mode(On, Of PHP_INI_ SYSTEM 默认 值 为 Of 启用 该 指令 表示 PHP 在 上 述 约束 条 
件 下 运行 
safe_ mode_ gid(On, Off) PHP_INI_SYSTEM 默认 值 为 Of 启用 安全 模式 时 ， 如 果 还 启用 了 


safe_ mode_gid(On.Of, 在 打开 文件 
时 ， 就 会 强制 完成 UID 检查 


safe_ mode include dir(string) PHP INIL SYSTEM 默认 值 为 NULL | 当 启 用 该 指令 时 ， 会 使 上 面 两 个 指 
令 在 指定 位 置 失效 ， 即 打开 指定 文 
件 夹 时 ， 将 忽略 UID/GID 检查 

safe_ mode exec dir(string) 启用 该 指令 , 会 限制 通过 exec0 函 数 
只 能 执行 指定 目录 中 的 可 执行 程序 


safe_ mode allowed env_vars(string) 启用 该 指令 会 限制 用 户 能 通过 PHP 
脚本 修改 操作 系统 的 变量 
safe_ mode protected_env_vars(string) 可 以 明确 防止 修改 某 些 环 境 变 量 
LD LIBRARY PATH 
open_basedir (string) 启用 该 指令 可 以 建立 一 个 基 目录 ， 
所 有 文件 操作 都 限制 在 此 目录 中 


disable_functions(string) 启用 该 指令 可 以 禁用 某 些 函数 
disable_classes(string) 启用 该 指令 可 以 禁用 某 些 类 
ignore_user_abort(off.on) PHP_INI ALL 默认 值 为 On 启用 该 指令 ， 可 以 使 服务 器 忽略 由 
ee 
的 会 话 中 止 
4. 数据 处 理 


外 部 变量 是 指 通过 一 些 外 部 源 传递 给 脚本 的 变量 。Get、Post、Cookie、 操 作 系统 和 服务 器 都 可 
以 提供 外 部 数据 。 可 以 使 用 指令 来 影响 外 部 变量 的 使 用 。 其 详细 信息 如 表 1-9 所 示 。 


表 1-9 数据 处 理 指令 
指 令 作 用 域 说 了 明 


Tegister_globals(On.Off) PHP_INI_ SYSTEM 默认 值 为 O 任 设置 Environment、Get、Post、Cookie、 
Server 变量 为 全 局 变量 


Tegister long arrays(On.Off) | PHP _ INI SYSTEM 默认 值 为 O 仓 该 指令 确定 是 否 继续 使 用 已 经 废弃 的 语法 


PHP INI SYSTEM 默认 值 为 On 设 定 通 过 Get 方法 传 入 变量 信息 与 可 执行 
文件 传递 参数 类 似 
PHP INI SYSTEM 默认 值 为 8SMB | 设 定 PHP 脚本 以 Post 传递 数据 量 的 值 


register arec arev(On.Off) 


post max size(integer)M 


指令 说 明 

magic_quotes gpc(On.0fD) PHP INI SYSTEM 默认 值 为 On 确定 是 否 对 Get、Post 和 Cookie 方法 传输 
的 数据 启用 魔法 引号 

magic_quotes_sybase(On.Off) | PHP INI ALL 默认 值 为 Of 此 参数 只 在 启用 magic_quotes_runtime 时 
有 效 ， 如 果 启用 了 magic_quotes_sybase， 
所 有 来 自 外 部 资源 的 数据 都 将 使 用 一 个 单 
引号 而 不 是 反 斜 线 进行 转 义 

auto_prpend_file(string) PHP INI SYSTEM 默认 值 为 NULL | 设置 在 PHP 文件 中 要 加 载 的 文件 名 和 相应 
路 径 

default_ mimetye(string) PHP_INI ALL 默认 值 为 设置 PHP 文件 的 类 型 

SAPI DEFAULT CHARSET 
default_charset(string) PHP_INI_ALL 默认 值 为 设置 PHP 文件 在 Content-type 首部 中 输出 


SAPI DEFAULT CHARSET 字符 编码 形式 。 默 认 情 况 下 是 iso-8859-1 


variables_order (string) PHP INI ALL 默认 值 为 NULL 该 指令 确定 Environment、Get、Post、 
ee | Cookie、Server 变量 的 解析 顺序 
arg_separatorinput (sring) && 是 Post 或 Get 方法 用 来 分 隔 输入 变量 的 
标准 字符 
arg_separator.output (sring) 能 自动 生成 URL， 并 使 用 标准 的 & 符 号 分 
隔 输入 变量 


5.， 路径 和 目录 
路 径 和 目录 指令 主要 用 来 设置 PHP 文件 的 默认 路 径 ， 这 些 路 径 用 于 导入 函数 库 和 扩展 包 ， 以 及 
确定 用 户 Web 目录 和 Web 文档 目录 。 其 详细 信息 如 表 1-10 所 示 。 
表 1-10 路 径 和 目录 指令 


指令 说 明 
include_path (string) 设 定 include0、require0 和 fopen_with_path0 等 函数 
PHP INCLUDE PATH 使 用 的 基本 路 径 ， 可 以 指定 多 个 目录 
doc root 此 参数 确定 提供 所 有 PHP 脚本 的 默认 位 置 , 此 参数 
PHP INCLUDE PATH 非 空 时 才 会 使 用 


user_dir(string) PHP_INI_SYSTEM 默认 值 为 用 来 指定 在 使 用 /~usemame 约定 打开 文件 时 ，PHP 
PHP INCLUDE PATH 使 用 的 绝对 目录 

extension_dir(string) | PHP_INI SYSTEM 默认 值 为 设置 PHP 可 加 载 的 扩展 包 的 位 置 ， 默 认 情况 下 为 ./ 
PHP EXTENSION DIR 

enable dl(On.Off) ”| PHP_INI SYSTEM 默认 值 为 On 允许 用 户 在 运行 时 加 载 PHP 扩展 包 


include_path 指令 是 一 个 比较 重要 的 指令 ， 在 不 同 的 系统 中 具有 不 同 的 写法 ， 如 格式 和 系统 的 
path 环境 变量 类 似 : 一 组 目录 的 列表 ， 在 UNIX 下 用 冒号 分 隔 ， 而 在 Windows 下 用 分 号 分 隔 。 


// Unix include path 

include path=".:/php/includes" 
// Windows include path 

include path=".;c:\php\includes" 
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在 包含 路 径 中 使 用 “.” 可 以 允许 相对 路 径 ， 它 代表 当前 目录 。 
6. 文件 上 传 
PHP 文件 支持 Post 方法 上 传 和 管理 文本 文件 及 二 进 制 文件 。 有 三 个 指令 支持 这 个 功能 ， 如 表 
1-11 所 示 。 
表 1-11 文件 上 传 指令 
指 令 作 用 域 说 有 明 
file_uploads (On.Of) PHP INI SYSTEM 默认 值 为 On 是 否 允许 HTTP 文件 上 传 
upload_tmp_dir(string) PHP INI SYSTEM 默认 值 为 NULL | 设 定 文件 上 传 时 存放 文件 的 临时 目录 
Upload _ max filesize(integeDJM | PHP_INI SYSTEM 默认 值 为 2MB 所 上 传 的 文件 的 最 大 大 小 


对 于 本 章 中 没有 介绍 的 配置 指令 ， 会 在 后 面 的 章节 中 陆续 提 到 。 


1.4 一 个 简单 的 PHP 例子 


本 节 将 使 用 一 个 PHP 的 简单 案例 ， 简 单 了 解 PHP， 为 后 面 学 习 PHP 打下 坚实 的 基础 。 
创建 一 个 案例 , 该 案例 显示 一 个 字符 串 信息 。 假 设计 算 机 上 已 经 安装 了 PHP 运行 所 需要 的 软件 。 
打开 记事 本 ， 输 入 下 列 代码 : 

案例 1-1 
<html> 
<head><title> 一 个 例子 </title></head> 
<body> 

<h1> 你 好 </h1> 

<?php 

echo ("这 是 第 一 个 PHP 例子 ") ; 
?> 
</body> 

</html> 


将 该 文件 保存 在 C:\Web\apache\htdocs 目录 下 ， 文 件 名 为 prphp。 打 开 正 浏览 器 ， 在 地 址 栏 中 
输入 http://localhost:8080/prphp， 会 显示 如 图 1-26 所 示 的 窗口 。 

从 上 面 的 代码 中 ， 可 以 看 出 PHP 代码 可 以 嵌入 HTML 
标记 语言 中 ， 婴 入 的 方式 是 使 用 <?php ?> 标记 ， 在 嵌入 标 Ee 
记 中 ， 使 用 输出 函数 输出 了 一 个 字符 串 信息 。 Cs 办 

如 果 要 开发 一 个 大 型 的 PHP 项 目 ， 不 可 能 把 代码 都 放 | 你 好 
到 htdocs 文件 夹 内 ,需要 建立 该 项 目的 文件 夹 。 如 要 建立 一 | -Tre 人 
个 myweb 项 目 , 可 以 直接 在 C:\Web\apache\htdocs 目录 下 建 医 
立 文件 夹 myweb, 把 该 项 目的 PHP 文件 放 到 里 面 就 可 以 了 。 图 1.26 PHP 显示 
如 果 要 建立 的 项 目 不 想 放 到 C:\Web\apache\htdocs 目录 的 下 
面 ， 想 在 另外 的 盘 符 建立 一 个 文件 来， 如 正 盘 的 test3 目录 ， 并 在 该 目录 下 放置 一 个 info.php 页 面 ， 
这 时 需要 打开 C:\Web\apache\conf 目录 下 的 httpd.conf 文件 ， 在 里 面 <Directory "C:/Web/apache/htdocs"> 


标记 的 上 面 添加 下 列 代码 


Alias /test3 "e:/test3" 
<Directory "es/test3"> 
Options Indexes FollowSymLinks 
AllowOverride None 
Order allow,deny 
Allow from all 
</Directory> 


然后 保存 httpd.conf 文件 ， 重 新 启动 Apache 服务 器 。 打 开 正 浏览 器 ， 在 地 址 栏 中 输入 
http://localhost:8080/test3/pr.php， 单 击 【 转 到 】 按 钮 ， 会 显示 如 图 1-27 所 示 的 窗口 。 
外 Fivtezt3 Yo rp ey 区 
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图 1-27 信息 显示 
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PHP 是 一 种 HTML 内 翌 式 语言 ，PHP 与 微软 的 ASP 颇 有 几 分 相似 ， 都 是 一 种 在 服务 器 端 执行 
并 嵌入 在 HTML 文档 的 脚本 语言 。PHP 独特 的 语法 混合 了 C、Java、Perl 以 及 PHP 自 创 的 新 语法 ， 
容易 被 初学 者 掌握 。PHP 具有 非常 强大 的 功能 ， 它 能 实现 所 有 的 CGI 或 者 JavaScript 的 功能 ， 而 且 
支持 几乎 所 有 主流 数据 库 以 及 操作 系统 。 本 章 将 对 PHP 的 基础 语法 进行 详细 的 说 明 。 


全 内容 摘要 |apstract 


掌握 嵌入 PHP 代码 的 几 种 方式 
掌握 PHP 注释 的 使 用 

掌握 PHP 程序 输出 语句 

掌握 PHP 常用 的 数据 类 型 
熟练 掌握 PHP 数据 类 型 的 转换 
掌握 PHP 变量 的 创建 

掌握 PHP 的 常用 操作 符 

熟练 掌握 PHP 的 控制 语句 
掌握 PHP 的 包含 语句 


2.1 PHP 脚本 基础 


在 学 习 PHP 语法 之 前 ,需要 了 解 创 建 PHP 脚本 代码 的 基础 知识 ,如 在 一 个 PHP 页 面 中 放 入 PHP 
脚本 、 使 用 PHP 注释、 输出 服务 器 端 信息 等 。 
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2.1.1 府 入 PHP 代码 


PHP 是 一 种 嵌入 HTML 语言 中 的 脚本 语言 。 如 果 要 将 嵌入 在 HTML 文档 中 的 PHP 脚本 程序 与 
其 他 语言 区 分 开 ， 如 CSS、JavaScript 等 ， 就 要 把 嵌入 的 PHP 脚本 语言 放置 到 特定 的 成 对 的 标记 内 。 
这 样 当 解析 一 个 PHP 文件 时 ， 会 寻找 相应 的 开始 和 结束 标记 ， 这 些 标记 告诉 PHP 解析 器 开始 和 停 
止 的 位 置 。 此 种 解析 方式 可 以 使 PHP 嵌入 到 各 种 不 同 的 文档 中 。 在 一 对 开始 和 结束 标记 之 外 的 内 容 
会 被 PHP 解析 器 忽略 。 大 多 数 情况 下 PHP 都 是 嵌入 在 HIML 文档 中 。 在 HTML 页 面 中 嵌入 PHP 
脚本 ， 常 用 的 有 4 种 方式 ， 如 表 2-1 所 示 。 


表 2-1 赃 入 方式 
标 记 标记 名 称 描 述 
PHP 页 面 使 用 最 多 的 标记 
以 脚本 形式 嵌入 代码 的 方式 
短 标记 使 用 此 标记 需要 设 定 
ASP 风格 标记 同 ASP 相似 ， 需 要 设 定 


上 述 4 个 标记 的 使 用 示例 如 下 所 示 : 


<?php echo ' 推 荐 使 用 这 种 标记 ' ; ?> 
<script language="php"> 
echo “该 写法 和 使 用 Javascript 比较 相似 '; 
</script> 
<? echo ' 短 标记 '; ?> 
<% echo "You may optionally use ASP-style tags"; $%> 


下 面 对 这 4 种 常用 的 方式 分 别 进行 介绍 。 
1. 默认 语法 
<?php ?> 标记 是 PHP 最 常用 的 一 种 标记 ， 也 是 推荐 使 用 的 标记 。 该 标记 直接 可 以 使 用 ， 不 需 
要 进行 设置 。 下 面 创 建 一 个 案例 ， 演 示 一 下 该 标记 的 使 用 : 
案例 2-1 
<html> 
<head><title> 常 用 标记 </title></head> 
<?php 
echo "<p align=center>"; 
print "推荐 使 用 该 标记 "; 
区 
</html> 


在 C:\Web\apache\htdocs 目录 下 建立 myweb 文件 夹 , 将 上 述 代 码 保存 在 myweb 文件 夹 中 , 文件 
名 为 test.php。 打 开 正 浏览 器 ， 在 地 址 栏 中 输入 http://localhost:8080/myweb/test.php， 单 击 【 转 到 】 
按钮 ， 会 显示 如 图 2-1 所 示 的 窗口 。 

2. 脚本 标记 

使 用 <script language="php"> ”</scrip 人 > 标记 存放 PHP 
代码 ， 是 PHP 嵌入 HTML 页 面 的 另外 一 种 形式 。 该 标记 
的 使 用 形式 和 在 HIML 页 面 中 使 用 JavaScript 代码 比较 相 
似 。 该 标记 不 需要 进行 设置 ， 就 可 以 直接 使 用 。 图 2-1 使 用 标记 

3 短 标记 

<? ?> 是 PHP 代码 嵌入 的 另外 一 种 形式 ， 这 是 一 种 不 太 常用 的 形式 ， 这 种 标记 称 为 短 标记 。 该 
标记 在 使 用 时 ， 可 能 会 和 XML 语法 冲突 。 短 标记 的 使 用 如 下 所 示 : 

<? 

echo “' 这 种 标记 是 短 标记 ' 7 


is 


推荐 使 用 说 标记 
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该 标记 可 以 在 php.ini 中 进行 设置 ， 是 开启 状态 还 是 关闭 状态 。 打 开 C:\Web\php 文件 夹 下 的 
php.ini 文件 ， 在 该 文件 的 第 77 行 有 如 下 语句 : 


在 上 述 代码 中 ， 可 以 设置 shor_ open_tag=Off， 这 样 短 标记 在 PHP 页 面 中 就 不 能 使 用 了 。 

4. ASP 风格 标记 

<% ”%> 标 记 中 也 可 以 放置 PHP 代码 ， 该 标记 和 在 ASP、JSP 中 嵌入 脚本 程序 的 标记 一 样 。 如 
果 读 者 非常 喜欢 这 种 格式 ， 可 以 进行 设置 。 该 标记 的 使 用 形式 如 下 所 示 : 


一 般 情况 下 该 标记 是 处 于 关闭 状态 的 。 打 开 C:\Web\php 文件 夹 下 的 php.ini 文件 ， 在 该 文件 的 
第 85 行 有 如 下 语句 : 


如 果 把 asp_tags 属性 的 值 设 置 为 On， 该 标记 就 可 以 直接 使 用 了 。 

在 一 个 PHP 页 面 中 , 可 以 处 理 复杂 一 些 的 逻辑 功能 或 者 增加 一 些 样式 设置 , 这 样 不 可 避免 地 就 
会 出 现 HTML 标记 和 PHP 脚本 程序 等 多 种 标记 同 在 一 个 页 面 的 情况 。 这 样 的 形式 在 PHP 页 面 中 也 
是 允许 的 。 下 面 创 建 一 个 案例 ， 演 示 在 PHP 页 面 中 使 用 多 种 代码 块 ， 如 下 所 示 : 


将 上 述 代码 保存 为 manyphp 文件 ， 保 存 位 置 在 C:\Web\apache\htdocswyweb 目录 下 。 打 开 正 


浏览 器 ， 在 地 址 栏 中 输入 http://localhost:8080/myweb/many.php， 单 击 【 转 到 】 按 钮 ， 会 显示 如 图 2-2 
所 示 的 窗口 。 


EFS 一 


图 2-2 使 用 多 种 代码 


2.1.2 ”注释 


在 自己 编写 的 程序 中 加 入 注释 是 一 个 好 的 习惯 。 注释 起 到 功能 说 明 的 作用 , 无 论 过 了 多 长 时 间 ， 
自己 或 者 别人 重新 查看 程序 ， 有 注释 的 程序 会 更 加 容易 掌握 和 修改 。 在 PHP 脚本 语言 中 ,PHP 支持 
C、C++ 和 UNIX Shell 风格 (Perl 风格 ) 的 注释 ， 分 单行 注释 和 多 行 注释 。 下 面 将 针对 这 几 种 注释 
分 别 介绍 ， 如 表 2-2 所 示 。 
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表 2-2 注释 


单行 C++ 语法 | 如 果 加 入 的 注释 没有 超过 一 行 ， 这 时 可 以 选择 | <?php 
使 用 单行 注释 。 因 为 这 种 注释 很 短 ， 没 有 必要 | ” // 这 里 输出 了 一 个 字符 串 
界定 这 种 注释 的 结束 。 这 里 的 单行 注释 和 C++ | 。 echo "大 家 好 " 
相同 ， 故 称 为 单行 C++ 注 释 ?> 
Shell 语法 PHP 也 支持 这 种 C++ 风格 的 单行 语法 <?php 
(Perl 风格 ) # 这 里 输出 了 一 个 字符 串 
echo "大 家 好 " 
?> 
译 行 C 语 通常 需要 有 一 些 详细 的 功能 描述 或 其 他 解释 | <?php 
内 容 ， 这 些 说 明 可 能 包括 多 行 ， 同 时 标注 注释 | /* 
的 开始 和 结束 以 上 的 代码 执行 书库 链接 功能 
需要 注意 的 是 MySQL 数据 库 的 用 
户 名 和 密码 
下/ 
?> 


#/ 


需要 注意 的 是 ， 多 行 注释 语法 对 于 根据 代码 生成 文档 尤其 有 用 ， 因 为 这 样 可 以 很 明确 地 区 分 出 
各 个 注释 ， 如 果 使 用 单行 语法 ， 则 很 难 做 到 如 此 方便 。 在 实际 的 使 用 过 程 中 ， 这 几 种 注释 可 以 根据 
需要 同时 出 现在 一 个 程序 中 。 
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2.1.3 输出 


PHP 技术 是 一 种 动态 网 页 技术 ， 需 要 从 客户 端 获得 信息 ， 进 行 处 理 ， 并 把 处 理 的 结果 返回 到 客 
户 端 , 从 而 达到 交互 目的 。 在 PHP 中 把 信息 输出 到 客户 端 , 通常 采用 echo0、print0、printtO、sprinttO 
函数 。 其 函数 信息 如 表 2-3 所 示 。 


表 2-3 输出 函数 


语法 格式 函数 描述 函数 示例 

boolean print(argument) pimtO 函 数 负责 为 用 户 提供 反 | <?php 
馈 , 能 显示 原始 字符 串 和 变量 。| print "<p> 我 喜欢 使 用 PHP</p>"; 
如 果 输 出 成 功 ， 该 方法 就 会 返 | ?> 
回 一 个 布尔 值 <?php 

$str=" 使 用 PHP"; 

print("<p> 我 喜欢 $str</p>"); 


?> 


void echo (string echo0 函 数 与 print0 函 数 的 作用 | <?php 
$argl [. string $...]) 相同 ， 都 是 用 来 向 客户 端 输出 | $hh=" 今 天 ":; 
信息 。 但 有 两 点 不 同 ， 首先 ， Skk=" 很 好 "; 
echo0 函 数 不 能 用 在 复杂 的 表 | echo Shh." 天 气 "Skk" 出 去 玩 吧 "?> 
达 式 中 ， 因 为 它 返 回 void， 而 
printO 函数 返回 一 个 boolean 
值 。 其 次 ,echo0 函 数 能 输出 多 
个 字符 串 
int printf (string $format 该 函数 与 print0 函 数 的 功能 相 | 见 下 面 详细 信息 
[, mixed $args [. mixed $...]]) | 同 ， 都 是 向 客户 端 输出 信息 。 
但 是 使 用 printf0 函 数 输出 的 字 
符 串 是 一 个 格式 化 过 的 字符 串 
sprintfO string sprintf (string $format | sprintf) 函数 和 printf0 函 数 一 | <?php 
[. mixed $args [. mixed $...]]) | 样 ， 都 是 输出 格式 化 后 的 信息 | $str=sprintf("%601.2f".43.2): 
echo($str): 
?> 


除了 表 中 函数 信息 和 示例 外 ， 还 有 几 个 地 方 需要 注意 ， 观 察 下 面 的 print 使 用 示例 ， 如 下 所 示 : 


<?php 
$str=" 使 用 PHP"; 
print ("<p> 我 喜欢 "."$str</p>"); 


在 上 述 代码 中 ,创建 了 一 个 变量 ， 只 不 过 这 里 使 用 “.”， 这 里 的 圆 点 表示 字符 串 连 接 符 。 
从 表 中 可 以 知道 printf0 函 数 的 语法 格式 如 下 所 示 : 


int printf (string $format [, mixed S$args [, mixed $...]]) 


可 以 看 出 ， 函 数 的 返回 值 是 一 个 整 型 数值 ， 表 示 字 符 串 的 长 度 。args 表示 指定 的 参数 ， 但 是 它 
的 输出 将 根据 format 进行 格式 化 。format 参数 可 以 对 输出 数据 进行 充分 的 控制 ,如 对 齐 方 式 、 精度、 
类 型 或 位 置 。 这 个 参数 由 5 部 分 组 成 ， 如 表 2-4 所 示 。 


表 2-4 参数 名 称 3 


可 选 ， 这 一 部 分 确定 达到 正确 的 字符 串 大 小 所 用 的 填充 字符 。 默 认为 空格 ， 也 可 
以 指定 其 他 填充 提示 符 〈 在 字符 前 加 一 个 单 引号 7 


第 2 个 | 对 齐 提 示 符 ”| 可 选 ， 这 一 部 分 确定 输出 是 左 对 齐 还 是 右 对 齐 。 默 认为 右 对 齐 ， 可 以 用 一 个 负 号 
设置 为 左 对 齐 
第 3 个 | 宽度 提示 符 ”| 可 选 ， 这 一 部 分 确定 此 函数 输出 的 最 少 字 符 数 第 
第 4 个 | 精度 提示 符 | 可 选 ， 确 定 应 显示 的 小 数位 数 ， 这 一 部 分 只 影响 浮 点 数 类 型 的 数据 2 
第 5 个 类 型 提示 符 ”| 这 一 部 分 确定 如 何 转 换 参数 
它 所 支持 的 类 型 提示 符 如 表 2-5 所 示 。 到 
表 2-5 格式 化 类 型 时 
类 型 描述 吕 
册 将 参数 认为 是 一 个 整数 ， 显 示 为 二 进 制 数 
%c 将 参数 认为 是 一 个 整数 ， 显 示 为 对 应 的 ASCII 字符 
册 将 参数 认为 是 一 个 整数 ， 显 示 为 有 符号 的 十 进 制 数 
好 将 参数 认为 是 一 个 浮 点 数 ， 显 示 为 浮 点 数 
Wo 将 参数 认为 是 一 个 整数 ， 显 示 为 八进制 数 
和 将 参数 认为 是 一 个 字符 囊 ， 显 示 为 字符 中 
册 将 参数 认为 是 一 个 整数 ， 显 示 为 无 符号 的 十 进 制 数 
本 将 参数 认为 是 一 个 整数 ， 显 示 为 小 写 的 十 六 进 制 数 
将 参数 认为 是 一 个 整数 ， 显 示 为 大 写 的 十 六 进 制 数 
下 面 创建 一 个 案例 ， 演 示 一 下 printf0 函 数 的 使 用 ， 案 例 代码 如 下 所 示 
案例 2-3 
<?php 


printf("(9.95*100)=%d", (9.95*100)); 
?> 
<?php 
$hh=2.12; 
printf ("%.3f", $hh); 
Ne 
<?php 
printf ("The %1\$s says: $2\$s, %2\$s","dog","bark"); 
?> 


将 上 述 代码 保存 在 C:\Web\apache\htdocsimyweb 目录 下 ， 文 件 名 为 aphp。 打 开 正 浏览 器 ， 在 
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地 址 栏 中 输入 http://localhost:8080/myweb/aphp， 单 击 【 转 到 】 
按钮 ， 会 显示 如 图 2-3 所 示 的 窗口 。 

从 运行 结果 中 可 以 看 出 , 第 一 个 输出 语句 在 格式 化 字符 串 
时 ,把 浮 点 数 转换 为 整 型 时 ， 进 行 舍 位 ， 并 把 舍 位 后 的 结果 输 zs6 
出 。 第 二 个 输出 语句 表示 输出 参数 的 指定 位 数 , 这 里 指定 的 输 | me eo sws: emk bart 
出 位 数 是 3。 第 三 个 输出 语句 表示 输出 的 是 字符 串 信息 ， 这 里 
需要 注意 的 是 %1\$ 表 示 输 出 的 是 第 一 个 参数 ， 依 次 类 推 。 图 2-3 ”字符 串 格式 化 输出 


(9. 95+100)=s94 


©@ f echo0 浮 数 和 printO 函 数 在 功能 上 可 以 互 换 ， 在 执行 速度 上 echo0 函 数 稍 快 一 些 。 


示 


2.2 ”数据 类 型 


数据 类 型 是 具有 相同 特性 的 一 类 数据 的 统称 ， 如 该 类 数据 在 内 存 中 占有 空间 的 大 小 相同 ， 使 用 
的 函数 相同 等 特性 。 PHP 从 诞生 到 现在 , 一 直 在 不 断 地 添加 新 的 数据 类 型 ,目前 PHP 共有 8 种 数据 
类 型 ， 如 表 2-6 所 示 。 


表 2.6 数据 类 型 
孝 据 类 型 类 别 
boolean 复合 类 型 
integer 复合 类 型 
float 特殊 类 型 
sting 特殊 类 型 


在 后 面 的 操作 中 可 能 会 遇 到 双 精 度 (double) 类 型 的 ， 它 和 float 类 型 相同 ， 由 于 历史 原因 ， 二 
者 同时 存在 。 为 了 确保 代码 的 易 读 性 ， 还 增加 了 一 些 伪 类 型 ， 如 number、mixed、callback。 本 节 将 
详细 地 介绍 常见 的 数据 类 型 ， 以 及 数据 类 型 之 间 的 相互 转换 等 。 


2.2.1 标量 数据 类 型 
标量 数据 类 型 只 能 包含 单一 的 数据 信息 ， 如 包含 了 整 型 数据 ， 就 不 能 包含 字符 串 信息 。 常 见 的 


标量 数据 类 型 有 布尔 型 、 浮 点 型 、 整 型 和 字符 串 型 。 其 使 用 信息 如 表 2-7 所 示 。 


表 2-7 标量 数据 类 型 
名 称 描 述 示 例 
布尔 型 ”| boolean 表达 了 真 值 ， 可 以 为 True 或 False， 这 两 个 


值 不 区 分 大 小 写 。 也 可 以 用 0 表示 False, 非 0 值 表 


示 Tme 


名 称 描 述 示 例 

整 型 ”| 不 含 小 数位 的 数 ， 一 个 integer 的 数 是 集合 Z={-…, | <?php 
-2, -1,0, 1, 2. …} 中 的 一 个 数 。 整 型 数 可 以 用 十 进 | Sa = 1234: // 十 进 制 数 : 
制 、 十 六 进 制 或 八进制 符号 指定 ， 并 且 前 面 可 以 加 | $a =_123:// 一 个 负数 : 
上 “+” 号 或 者 “-” 号 。 如 果 用 八进制 符号 ， 数 字 | $a = 0123: // 八进制 数 等 于 十 进 制 的 83) Ee 


前 必须 加 上 0〈 零 )， 如 果 用 十 六 进 制 符 号 ,数字 前 
必须 加 上 0x 


浮 点 型 | 浮 点 数 〈floating-point number) 也 称 为 单 精度 数 
(float)、 双 精度 数 (double) 或 实数 (real number)， 
可 以 指定 包含 小 数 部 分 的 数 。 浮 点 数 用 于 表示 货币 
值 、 重 量 、 距 离 ， 以 及 用 简单 的 整数 无 法 满足 要 求 
的 其 他 表示 


字符 串 | 字符 串 是 一 系列 字符 的 组 合 ， 字 符 串 的 使 用 可 以 使 
用 下 面 几 种 方式 来 定义 ， 分 别 为 : 单 引号 、 双 引号 、 
定 界 符 等 


注意 ， 整 型 数 的 字 长 和 平台 有 关 ， 通常 最 大 值 大 约 是 20 亿 (32 位 有 符号 数 )。PHP 不 支持 无 符 
号 整数 。 如 果 给 定 的 数值 超出 了 integer 的 范围 ,将 会 被 解释 为 Boat。 同 样 如 果 执 行 的 运算 结果 超出 
了 integer 范围 ， 也 会 返回 float。PHP 中 没有 整除 的 运算 符 ，1/2 产生 出 float 类 型 0.5。 浮 点 数 的 字 
长 和 平台 有 关 ， 通 常 最 大 值 是 1.8e308， 并 具有 14 位 十 进 制 数 字 的 精度 (64 位 IEEE 格式 )。 

创建 一 个 字符 串 ， 最 简单 的 方法 是 用 单 引 号 (字符 ') 括 起 来 。 在 字符 串 中 要 表示 一 个 单 引号 ， 
需要 用 反 斜 线 (\) 转 义 ， 和 很 多 其 他 语言 一 样 ， 如 果 在 单 引号 之 前 或 字符 串 结 尾 需 要 出 现 一 个 反 斜 
线 ， 需 要 用 两 个 反 斜 线 表 示 。 注 意 ， 如 果 试 图 转 义 任何 其 他 字符 ， 反 斜 线 本 身 也 会 被 显示 出 来 ， 所 
以 通常 不 需要 转 义 反 斜 线 本 身 。 使 用 单 引 号 创建 的 字符 串 如 下 所 示 。 

<?php 

echo “' 这 是 一 个 简单 的 字符 串 '; 


echo “' 看 这 个 字符 串 : "I\'11] be back"';// 输 出 的 结果 是 “I']11 be back” 
echo “' 请 删除 C:\\*.*? 文 件 '; // 输 出 格式 为 : 请 删除 C:\*.*? 文 件 


$a 二 0x1A;// 十 六 进 制 数 〈 等 于 十 进 制 的 26) 
> 

<Iphp 

$a= 1.234; 

$a= 1.2e3; 

$a= 7E-10: 

?> 


见 下 面 内 容 
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?> 


创建 一 个 字符 串 ， 也 可 以 用 双 引 号 (字符 ") 把 字符 括 起 来 。 在 这 种 字符 串 格式 中 ， 可 以 使 用 更 
多 的 转 义 字符 。 常 用 的 转 义 字符 如 表 2-8 所 示 。 


表 2-8 转 义 字符 
序 列 含义 
un 换行 (LF 或 ASCII 字符 0x0A (10)) 
Vv 回 车 (CR 或 ASCII 字符 0x0D (13)) 
vt | 水 平 制 表 符 (HT 或 ASCIT 字符 0x09 (9)) 
\ | 反 斜 线 
' | 美元 符号 
V | 双 引 号 
\[0-7]{1.3} | 此 正则 表达 式 序列 匹配 一 个 用 八进制 符号 表示 的 字符 


Wx[0-9A-Fa- 各 {1.2} | 此 正则 表达 式 序列 匹配 一 个 用 十 六 进 制 符号 表示 的 字符 
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此 外 ， 如 果 试 图 转 义 任何 其 他 字符 ， 反 斜 线 本 身 也 会 被 显示 出 来 。 双 引号 字符 串 最 重要 的 一 点 
是 其 中 的 变量 名 会 被 变量 值 蔡 代 。 使 用 双 引 号 创建 字符 串 的 示例 如 下 所 示 : 


$a=" 中 国 ， 你 好 "; 


外 一 种 创建 字符 串 的 方法 是 使 用 定 界 符 “<<<”， 应 该 在 <<< 之 后 提供 一 个 标识 符 ， 然 后 是 字 
符 串 ， 最 后 是 同样 的 标识 符 结束 字符 串 。 结 束 标识 符 必 须 从 行 的 第 一 列 开 始 。 同 样 ， 标 识 符 也 必须 
遵循 PHP 中 其 他 任何 标签 的 命名 规则 : 只 能 包含 字母 、 数 字 、 下 划 线 ， 而 且 必 须 以 下 划 线 或 非 数字 
字符 开始 。 定 界 符 文本 和 双 引 号 字符 串 一 样 ， 只 是 没有 双 引 号 ， 这 意味 着 在 定 界 符 文本 中 不 需要 转 
义 引 号 ， 但 是 仍然 可 以 用 以 上 列 出 的 转 义 代码 。 使 用 定 界 符 创 建 字 符 串 的 示例 如 下 所 示 : 

$str = <<<EOD 

在 这 里 我 们 采用 了 定 界 符 创 建 字 符 串 ， 

该 字符 串 是 一 个 多 行 字符 串 


EOD; 


在 上 述 代 码 中 ,“<<<” 符 号 表示 一 个 定 界 符 ，EOD 表示 一 个 标识 符 ， 标 识 符 中 间 是 字符 串 信 
息 。 


2.2.2 ”复合 数据 类 型 
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复合 数据 类 型 就 是 把 多 种 相同 的 类 型 信息 组 合 在 一 起 ， 如 将 数组 (array) 和 对 象 (object) 组 
合 在 一 起 ， 主 要 用 来 解决 一 个 实体 具有 多 个 数据 的 情况 。 其 详细 信息 如 表 2-9 所 示 。 


表 2-9 复合 数据 类 型 


名 称 示例 

数组 (array) | 数组 是 一 种 能 够 存放 多 个 相同 类 型 数据 的 数据 结构 。 如 可 以 在 一 | $a[0]=67; 
个 数组 中 放置 多 个 字符 串 数值 ， 或 者 多 个 整 型 值 。 在 PHP 中 ， | sa[1]=78: 
数组 实际 上 就 是 一 个 有 序 图 ， 数 组 中 的 值 (values) 有 序 地 排列 ，| $a[" 刘 海松 "]-67: 
数组 中 存放 的 值 可 以 通过 数组 的 排列 号 码 (keys) 加 上 数组 名 称 | Sa* 刘 红 霞 中 -78: 
来 获得 

对 象 《objeeb | 对 象 是 由 类 生成 的 ， 在 对 象 中 可 以 包含 方法 和 变量 。 一 个 类 可 以 | <?php 
创建 不 同形 式 的 对 象 ,对象 是 面向 对 象 语言 的 核心 概念 。 在 PHP | class foo 


中 ， 创 建 一 个 对 象 需要 显 式 声明 ， 要 创建 一 个 对 象 首 先 要 创建 一 
个 类 ， 在 类 中 可 以 声明 变量 和 方法 


{ 
function do_foo0 
echo "Doing foo.": 
} 
} 
S$bar = new foo: 
$bar->do_foo0: 
?> 


对 于 对 象 和 数组 的 其 他 知识 点 ， 会 在 后 面 的 章节 中 陆续 提 到 。 


2.2.3 ”特殊 数据 类 型 


在 PHP 中 ,除了 上 面 常 用 的 数据 类 型 外 ,还 存在 一 些 用 在 特殊 方面 的 数据 类 型 , 如 资源 (resource) : 
和 空 (NULL) 数据 类 型 。 其 详细 信息 如 表 2-10 所 示 。 i 


表 2-10 ”特殊 数据 类 型 

名 称 描述 

资源 一 种 特殊 的 数据 类 型 ， 是 保存 了 到 外 部 资源 的 一 个 引用 。 在 
PHP 中 ，PHP 页 面 可 能 要 获取 不 同 的 资源 ， 如 数据 库 资 源 、 文 
件 资源 、 网 络 资源 、 图 像 资 源 等 、 这 时 就 需要 建立 一 个 指针 来 
指向 这 些 资源 ， 在 使 用 时 直接 调用 指针 就 可 以 了 。 这 些 指针 保 
存 着 对 这 些 资源 的 连接 ， 直 到 使 用 资源 的 程序 结束 ， 指 针 才 不 
指向 资源 ， 即 撤销 资源 。 资 源 是 通过 专门 的 函数 来 建立 和 使 用 
的 
NULL ”| 表示 空 或 者 无 ， 什 么 也 没有 。NULL 不 表示 空格 ， 不 表示 零 ， 
它 表示 没有 值 一 个 变量 在 下 列 几 种 情况 下 ,会 被 认为 是 NULL 
值 : 被 赋值 为 NULL、 没 有 被 赋值 、 使 用 unsetO 函 数 清除 等 情 
况 。NULL 数据 类 型 只 有 一 个 值 NULL 


2.2.4 类 型 强制 转换 


前 面 介绍 了 在 PHP 中 常用 的 数据 类 型 ， 以 及 每 一 种 数据 类 型 的 使 用 方法 。 那 么 这 些 数据 类 型 之 
间 是 否 可 以 相互 转换 ， 如 把 一 个 浮 点 型 变量 转换 成 一 个 整 型 变量 ， 把 一 个 字符 串 变 量 转换 为 布尔 类 
型 变量 。 在 PHP 中 ， 可 以 把 一 种 数据 类 型 强制 转换 成 另外 一 种 数据 类 型 ， 这 称 为 强制 类 型 转换 。 转 
换 过 的 变量 ， 就 以 新 的 类 型 进行 计算 。 转 换 的 格式 是 在 要 转换 的 变量 前 面 加 上 要 转换 的 类 型 ， 其 形 
式 如 表 2-11 所 示 。 


示 例 


$result = mysql connect 


("localhost", "username", "pass"); 
//$result 是 一 个 资源 类 型 的 变量 
Print $result; 


<?php 
$var=NULL: 


?> 
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表 2-11 类 型 转换 符 

转换 操作 符 转换 为 
(array) (object) 对 象 
(bool) 或 (boolean) (real) 或 (double) 或 (floab 浮 点 数 
(inb 或 (integeD) (string) 字符 串 
下 面 创建 一 个 案例 ， 演 示 一 下 强制 类 型 的 转换 ， 如 下 列 代码 所 示 : 
案例 2-4 
<?php 

$i=32; 


$ii=(double) $i; 
echo ($ii); 
echo("<br>"); 
$a=2.57; 
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$aa= (int)$a; 

print ($aa); 

echo("<br>"); 

$s=" 你 好 "; 

$ss=(int)$s; 

$sss=(object)$s; 

print ($ss); 

echo ("<br>"); 

print $sss->scalar; 
?> 


将 上 述 代 码 保存 ， 文 件 名 为 convertphp， 文 件 保存 的 位 置 在 C:\Web\apache\htdocs\myweb 目录 
下 ， 打 开 正 浏览 器 ， 在 地 址 栏 中 输入 http://localhost:8080/myweb/convert.php， 单 击 【 转 到 】 按 钮 ， 
会 显示 如 图 2-4 所 示 的 窗口 。 TFTRTPTT 

从 执行 结果 中 可 以 看 出 , 可 以 把 一 个 整 型 转换 为 浮 点 | mw 是 四 
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型 ,也 可 以 把 一 个 浮 点 型 转换 为 整 型 ， 当 从 数据 类 型 范围 
大 的 向 数据 类 型 范围 小 的 转换 时 ,会 舍弃 一 部 分 数值 ， 如 
案例 中 从 浮 点 型 到 整 型 转换 时 把 2.57 小 数位 舍弃 了 。 当 
把 一 个 字符 串 类 型 转换 为 整 型 时 , 会 返回 一 个 0 值 。 任何 
一 种 数据 类 型 都 可 以 转换 为 对 象 ， 转 换 后 ,就 成 为 该 对 象 
的 一 个 属性 ， 属 性 名 为 scalar， 可 以 通过 对 象 名 引用 该 属性 。 


2.2.5 “类 型 自动 转换 


图 2-4 类 型 强制 转换 


在 PHP 中 ,还 有 一 种 类 型 转换 的 情况 ， 即 类 型 的 自动 转换 。 这 种 转换 是 根据 变量 所 处 的 环境 来 
转换 的 ， 将 变量 转换 为 最 适合 的 类 型 。 究 其 原因 ，PHP 是 一 种 弱 类 型 的 语言 。 创 建 一 个 案例 ， 演 示 
弱 类 型 的 使 用 情况 ， 其 代码 如 下 所 示 : 


案例 2-5 

<?php 
$value=5; 
$count="15"; 
$value+=$count; 
echo ($value); 
echo("<br>"); 
$s1="24 个 人 "; 
$s2=10; 
$sl=$s2+$s1; 
echo($s1); 
echo ("<br>"); 
$total="1.0"; 
if($total) 

echo "这 是 一 个 布尔 值 "; 


echo ("<br>"); 


$vall="1.2e3"; 

$val2=2; 

echo $vall*$val2; 
> 


将 上 述 代码 保存 ， 文 件 名 为 convertl php， 文 件 保存 在 C:\Web\apache\htdocsmyweb 目录 下 ， 如 
果 以 后 文件 的 保存 位 置 不 再 特别 说 明 ， 都 是 保存 在 该 位 置 下 。 打 开 正 浏览 器 ， 在 地 址 栏 中 输入 
http://localhost:8080/myweb/convertl.php， 单 击 【 转 到 】 按 
钮 ， 会 显示 如 图 2-5 所 示 的 窗口 。 

从 执行 结果 中 可 以 看 出 , 如 果 整 型 变量 $value 和 包含 
数值 的 字符 串 相 加 ， 该 字符 串 会 自动 转换 为 整 型 变量 相 
加 。 如 果 一 个 字符 串 的 开始 位 置 包含 了 数值 , 那么 再 和 整 
型 的 变量 相 加 , 结果 同样 是 一 个 整 型 值 。total 变量 是 一 个 
整 型 变量 , 但 是 当 用 到 条 件 语句 中 时 就 会 自动 转换 为 布尔 站 
类 型 ， 该 值 为 True 值 。 在 开发 PHP 页 面 时 ， 应 注意 这 种 情况 的 发 生 。 


2.2.6 与 类 型 有 关 的 函数 


PHP 是 一 种 弱 类 型 的 语言 ， 声 明 变 量 时 ， 可 以 不 带 数 据 类 型 。 如 果 在 PHP 的 操作 中 ， 需 要 获取 
某 个 变量 的 数据 类 型 或 者 设置 某 个 变量 的 数据 类 型 ， 这 时 就 要 用 到 两 个 函数 ， 分 别 为 settypeO 函 数 
和 gettype0 函 数 。 其 详细 信息 如 表 2-12 所 示 。 


表 2-12 类 型 函数 
名 称 | 语法 格式 函数 说 明 使 用 示例 


settype0 | bool settype 将 变量 var 的 类 型 设置 成 指定 的 数据 类 型 | <?php 

(mixed $var. type。type 的 可 能 值 为 : boolean (或 为 bool, | $foo="5bar":  // string 

string $type) ”| 从 PHP 4.2.0 起 )、integer (或 为 int, 从 PHP | $bar =true:  //boolean 

4.2.0 起 )、float (只 在 PHP 4.2.0 之 后 可 以 使 | settype($foo，"integer"); // $foo 现在 
用 , 对 于 旧版 本 中 使 用 的 double 现 已 停 用 )、| 是 5 (integer) 
string、array、object、NULL (从 PHP 4.2.0 settype(Sbar "string"): // $bar 现在 
起 )。 函 数 如 果 成 功 则 返回 True， 否 则 返回 | 是 “1” (string) 

False 7> 


以 字符 串 的 形式 返回 指定 变量 的 数据 类 型 。 | 略 
返回 的 数据 类 型 也 是 8 个 ， 和 settype0 函 数 
中 设置 的 数据 类 型 完全 一 致 


注意 ， 不 要 使 用 gettypeO 函 数 来 测试 某 种 类 型 ， 因 为 其 返回 的 字符 串 在 未 来 的 版 本 中 可 能 需要 
改变 。 此 外 ， 由 于 包含 了 字符 串 的 比较 ， 它 的 运行 速度 较 慢 ， 可 使 用 is_*O 函 数 代替 。 


2.2.7 ”类 型 标识 符 函 数 


gettypeO | string gettype 
(mixed $var) 


在 前 面 的 小 节 中 ,介绍 了 可 以 获取 和 设置 某 个 变量 的 数据 类 型 。 如 果 要 判断 某 个 变量 的 数据 类 
型 ,那么 需要 使 用 到 另外 一 组 函数 ， 类 型 标识 符 函 数 。 这些 函数 主要 用 来 判断 某 些 变量 的 数据 类 型 ， 
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分 别 为 is_arrayO、is_boolO0、is_floatO0、is_int0、is_null0、is_numericO、is_objectO 、is_resourceO、 
is_scalar0 和 is_string0， 所 有 这 些 函 数 都 有 相同 的 命名 约定 、 参 数 和 返回 值 。 其 详细 信息 如 表 2-13 
所 示 。 


表 2-13 标识 符 函数 

函数 名 称 语法 格式 功 能 
is arrayO bool is array (mixed $var) 检测 变量 是 否 是 数组 , 如 果 var 是 array 则 返回 True, 否则 返回 False 
is_bool0 bool is bool (mixed $var) 检测 变量 是 否 是 布尔 型 ， 如 果 var 是 boolean 则 返回 Trme， 否 则 返 
回 False 

is float| bool is float (mixed $var) 检测 变量 是 否 是 浮 点 型 ， 如 果 var 是 float 则 返回 Tme， 否 则 返回 
False 

is_intO bool is_ int (mixed $var) 检测 变量 是 否 是 整数 ， 如 果 var 是 integer 则 返回 True， 和 否则 返回 
False 


is_null0 检测 变量 是 否 为 NULL, 如 果 var 是 NULL 则 返回 True, 否则 返回 
False 

is numericO 检测 变量 是 否 为 数字 或 数字 字符 串 ， 如 果 var 是 数字 或 数字 字符 串 
则 返回 True， 否 则 返回 False 

is_objectO 检测 变量 是 否 是 一 个 对 象 ， 如 果 var 是 一 个 object 则 返回 True， 否 
则 返回 False 

is_resource() 检测 变量 是 否 为 资源 类 型 ， 如 果 var 是 resource， 则 返回 Trme， 否 
则 返回 False 

is_scalar0 检测 变量 是 否 是 一 个 标量 ， 如 果 var 是 一 个 标量 ， 则 返回 True， 否 
则 返回 False 

is_string() 检测 变量 是 否 是 字符 串 ， 如 果 var 是 string 则 返回 Tme， 否 则 返回 
False 


表 2-13 中 的 函数 形式 比较 相似 ， 这 里 只 举 一 个 示例 ， 如 下 所 示 : 


<2php 
$a = False; 
$b = 0; 


if (is_bool($a)) {// 因 为 a 是 布尔 型 ， 所 以 结果 为 真 
print "Yes，this is a boolean"7 

1 

if (is _bool($b)) {// 因为 b 不 是 布尔 型 ， 所 以 结果 为 非 真 


print "Yes，this is a boolean"7 


2 


变量 是 一 个 用 来 临时 存储 数值 的 指针 ， 一 个 变量 具有 下 面 几 个 特性 : 变量 的 名 称 、 变 量 的 数据 
类 型 、 变 量 的 作用 域 范围 等 。 其 中 ， 变 量 的 名 称 表示 变量 存储 的 位 置 ， 变 量 的 数据 类 型 表示 变量 的 


大 小 ; 变量 的 值 是 临时 的 ， 当 程序 运行 时 ， 该 值 是 存在 的 ， 如 果 程 序 结束 ， 变 量 的 值 就 会 丢失 。 
2.3.1 变量 的 命名 


标识 符 是 一 个 标识 不 同 对 象 的 符号 , 如 变量 的 名 称 、 函 数 的 名 称 , 或 者 用 户 自 定义 对 象 的 名 称 。 
在 PHP 中， 标识 符 的 命名 必须 符合 下 面 的 规定 : 

e 标识 符 可 以 由 一 个 或 多 个 字符 组 成 , 但 必须 以 字母 或 下 划 线 开头 。 此 外 ,标识 符 只 能 由 字母 、 
数字 、 下 划 线 和 127~255 的 其 他 ASCI 字符 组 成 。 如 my a、Ss、_value 这 些 标 识 符 名 称 都 
是 合法 的 ， 而 gd"a、4t 这 些 变 量 的 名 称 是 不 合法 的 。 

。 标识 符 区 分 大 小 写 。 变量 recipe 不 同 于 变量 Recipe、IEciPe 或 recipE。 

。 标识 符 可 以 是 任意 长 度 。 这 样 程序 员 就 能 通过 标识 符 名 准确 地 描述 标识 符 的 用 途 。 

。 标识 符 名 称 不 能 与 任何 PHP 预定 义 的 关键 字 相 同 。 


2.3.2 ”创建 变量 


在 PHP 中 , 创建 一 个 变量 首先 要 定义 变量 的 名 称 。 变 量 名 称 区 分 大 小 写 ,变量 总 是 以 美元 符号 
$ 开 头 ， 然 后 是 变量 名 。 变 量 名 遵循 标识 符 的 命名 规则 ， 如 $Num 2、$a2、$_ug8 等 变量 的 名 称 是 合 
法 的 。 在 变量 的 创建 中 ，$Student 和 $student 是 两 个 不 同 的 变量 。 

在 创建 变量 的 过 程 中 ， 先 声明 变量 ,再 给 变量 赋值 是 一 个 好 的 习惯 。 在 PHP 中 不 需要 显 式 声明 
变量 ， 这 与 Perl 不 同 。 相 反 ， 变 量 声明 可 以 与 赋值 同时 进行 。 但 是 ， 可 以 这 样 做 并 不 意味 着 就 应 当 
这 样 做 。 声 明 变 量 的 示例 如 下 所 示 : 

$model; 

$_4a; 


在 PHP 中 , 给 变量 赋值 有 两 种 方式 ,分 别 为 值 赋 值 和 引用 赋值 。 值 赋值 是 直接 把 一 个 数值 通过 
赋值 表达 式 复 制 给 变量 ， 会 把 该 变量 原来 的 数值 覆盖 。 值 赋值 是 一 种 常用 的 变量 赋值 方法 ， 使 用 示 
例如 下 所 示 : 

$color="red"; 

$number=12; 

$age=12; 

$sum=12+"15"; 

在 PHP 4.0 中 ， 引 入 了 给 变量 赋值 的 另外 一 种 形式 ， 即 引用 赋值 ， 引 用 赋值 表示 所 创建 的 变量 
与 男 一 个 变量 引用 的 内 容 相同 。 因 此 ， 如 果 多 个 变量 引用 了 同一 个 内 容 ， 修 改 其 中 任意 一 个 变量 ， 
在 其 余 的 变量 上 都 将 有 所 反映 。 在 “=” 号 后 面 加 一 个 “& ”号 就 可 以 完成 引用 赋值 。 引 用 赋值 的 示 
例 形式 如 下 所 示 : 

S$valuel="Hello"; 


$value2=& $valuel; 
$value2="Goodbye"; 


在 上 述 代码 中 ， 创 建 一 个 变量 valuel1， 并 赋值 为 “Hello”， 变 量 $value2 采用 了 引用 赋值 ， 即 把 
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valuel 的 值 赋 给 了 value2， 此 时 这 两 个 变量 就 是 一 个 生命 共同 体 了 ， 当 一 个 发 生变 化 ， 另 外 一 个 就 
会 显示 出 相应 的 结果 。 

对 于 引用 赋值 还 可 以 采用 另外 一 种 形式 ， 把 将 “人 ”号 放 在 所 引用 变量 的 前 面 。 其 使 用 形式 如 
下 所 示 : 

$fo0 = 'Bob'; 

Sbar = &$foo; 

$bar = "My name is "7 


引用 传递 数值 很 重要 ， 在 后 面 的 章节 中 ， 还 会 涉及 到 这 方面 的 内 容 。 


2.3.3 ”变量 作用 域 


变量 的 作用 域 是 指 变量 在 程序 中 的 能 够 产生 影响 的 空间 范围 ， 在 这 个 空间 范围 内 该 变量 的 数值 
是 生效 的 。 在 PHP 页 面 中 ,任何 一 个 位 置 都 可 以 创建 变量 , 但 是 , 创建 变量 的 位 置 会 影响 变量 的 作 
用 域 范围 。PHP 变量 的 作用 域 范围 共有 4 个 ， 分 别 为 局 部 变量 、 函 数 参数 、 全 局 变量 、 静 态 变量 。 
其 详细 信息 如 表 2-14 所 示 。 


表 2-14 作用 域 
作用 域 范围 示 例 
局 部 变量 在 函数 中 声明 的 变量 可 以 认为 是 局 部 变量 , 即 它 只 能 在 该 函数 中 引 | <?php 
用 。 如 果 在 函数 外 赋值 ， 将 被 认为 是 完全 不 同 的 另 一 个 变量 〈 即 不 | $a=" 天 气 热 了 "; 
同 于 函数 中 所 包含 的 那个 变量 ) function getAO{ 


$a=" 我 喜欢 蓝 色 的 大 海 ": 
Print "$a"; 

} 

getAO; 

Print "$a"; 

?> 

<2?php 

function xl1($value){ 
S$value=$value*10; 


函数 参数 有 时 函数 会 带 有 相应 的 参数 ， 用 来 接收 外 部 传递 过 来 的 数值 。 在 函 

数 调用 完毕 后 ， 这 些 函数 参数 所 具有 的 值 就 会 消失 ， 可 以 得 出 ， 函 

数 参 数 的 应 用 范围 是 整个 函数 ， 和 函数 的 存在 有 直接 的 关系 
Tetum $value: 

} 

$v=x1(3): 

echo($v): 


全 局 变量 变量 的 作用 域 范围 是 整个 PHP 页 面 ， 即 在 PHP 页 面 中 的 任何 一 个 
函数 中 都 可 以 使 用 该 变量 ， 但 是 在 使 用 前 ， 必 须 声明 该 变量 是 全 局 
的 。 只 要 在 变量 前 面 加 上 关键 字 global， 就 可 以 将 其 识别 为 全 局 变 
量 。 如 果 将 global 关键 字 放 在 一 个 已 有 的 变量 前 面 ， 则 是 告诉 PHP 
要 使 用 同名 的 变量 


global $a. $b: 
$b = $a + $b: 
} 
SumO; 
echo $b;// 输 出 结果 为 3 
?> 
静态 变量 静态 变量 仅 在 局 部 函数 域 中 存在 ， 但 当 程序 执行 离开 此 作用 域 时 ， | <?php 
其 值 并 不 丢失 。 当 再 次 调用 该 函数 时 ， 还 能 保留 刚才 的 数值 。 在 变 | function Test0 
量 名 的 前 面 加 上 static 关键 字 就 能 够 创建 一 个 静态 变量 


static $a = 0; 
echo $a: 
$att; 


注意 ， 退 出 声明 变量 的 函数 时 ， 该 变量 及 相应 的 值 就 会 撤销 。 局 部 变量 很 有 用 ， 它 消除 了 出 现 
意外 副作用 的 可 能 性 ， 否 则 ， 这 些 副 作用 将 导致 可 全 局 访问 的 变量 被 有 意 或 无 意 地 修改 。 
声明 全 局 变量 的 另外 一 种 方式 是 使 用 PHP 的 GLOBALS[] 数 组 ， 该 方法 的 使 用 示例 如 下 所 示 : 


<?php 
Sa = 1; 
$b = 2; 


function Sum() 
{ 
$GLOBALS['b'] = $GLOBALS['a'] + $GLOBALS['b']; 
} 
Sum(); 
echo $b; 
Ea 


在 GLOBALS[] 数 组 中 ， 每 一 个 变量 为 一 个 元 素 ， 键 名 对 应 变量 名 ， 值 对 应 变量 的 内 容 。 


GLOBALS[] 之 所 以 在 全 局 范围 内 存在 ， 是 因为 GLOBALS[] 是 一 个 超 全 局 变量 。 
可 以 利用 静态 变量 来 计算 某 一 个 函数 被 调用 的 次 数 。 


2.3.4 可 变 变 量 

创建 一 个 变量 , 需要 给 该 变量 赋 一 个 固定 的 数值 . 那么 能 不 能 把 变量 的 值 作 为 一 个 变量 来 看 待 ， 
一 个 变量 的 变量 名 可 以 动态 地 设置 和 使 用 。 创 建 一 个 普通 的 变量 的 示例 如 下 所 示 : 

<?php 


$a = "hello'7 
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一 个 可 变 变量 获取 了 一 个 普通 变量 的 值 作为 这 个 可 变 变 量 的 变量 名 。 在 上 面 的 例子 中 将 hello 
上 两 个 美元 符号 〈$) 以 后 ， 就 可 以 作为 一 个 可 变 变量 的 变量 了 。 例 如 : 


<?php 
$$a = 'world'; 


使 


> 


这 时 ， 两 个 变量 都 被 定义 了 : $a 的 内 容 是 “hello”， 并 且 $hello 的 内 容 是 “world”。 因此 ， 可 以 
表述 为 : 
<?php 


echo "$a ${$a}™; 
?> 


以 下 写法 更 准确 并 且 会 输出 同样 的 结果 : 


<?php 
echo "$a $hello"; 


> 


它们 都 会 输出 : hello world。 


常量 的 含义 和 变量 的 含义 是 相对 的 。 变 量 的 值 在 程序 的 执行 过 程 中 会 发 生变 化 ， 而 常量 的 值 在 
PHP 页 面 脚本 程序 的 执行 中 不 发 生变 化 ， 并 且 不 能 修改 ， 如 常见 的 圆周 率 值 、 自 然 对 数值 等 。 常 量 
默认 为 区 分 大 小 写 。 按 照 惯例 ， 常 量 标识 符 总 是 大 写 的 。 常 量 名 和 其 他 任何 PHP 标记 遵循 相同 的 命 
名 规则 。 合 法 的 常量 名 以 字母 或 下 划 线 开始 ， 后 面 跟着 任何 字母 、 数 字 或 下 划 线 。 常 量 的 范围 是 全 
局 的 ， 可 以 在 脚本 的 任何 地 方 访问 常量 。 

可 以 用 define0 函 数 来 定义 常量 。 一 旦 定义 了 常量 ,就 不 能 再 改变 或 者 取消 定义 。 常 量 只 能 包含 
标量 数据 (boolean、integer、float 和 string)。 不 要 定义 resource 常量 。 可 以 简单 地 通过 指定 其 名 字 
来 取得 常量 的 值 ， 不 要 在 常量 前 面 加 上 $ 符 号 。 如 果 常 量 名 是 动态 的 ， 也 可 以 用 函数 constant0 来 读 
取 常 量 的 值 。 用 get_defined_constants0 函 数 可 以 获得 所 有 已 定义 的 常量 列表 。 使 用 defineO 函 数 创建 
常量 的 示例 如 下 所 示 : 


<?p] 
// 合 法 的 常量 名 

define ("FOO", "something"); 
define ("FOO2", "something else"); 
define ("FOO BAR", "something more"); 
// 非 法 的 常量 名 
define ("2FO0", "something"); 

?2> 


在 上 述 代码 中 , 使 用 define0 函 数 创建 常量 , 第 一 个 参数 表示 常量 的 名 称 ,第 二 个 参数 表示 常量 


的 值 。 常 量 和 变量 的 区 别 如 下 所 示 : 

。 常量 前 面 没有 美元 符号 ($)。 

。 常量 只 能 用 defineO 函 数 定义 ， 不 能 通过 赋值 语句 定义 。 

。 常量 可 以 不 用 理会 变量 范围 的 规则 而 在 任何 地 方 定义 和 访问 。 

。 常量 一 旦 定义 就 不 能 被 重新 定义 或 者 取消 定义 。 

。 常量 的 值 只 能 是 标量 。 

PHP 向 它 运 行 的 任何 脚本 提供 了 大 量 的 预定 义 党 量 .不 过 很 多 常量 都 是 由 不 同 的 扩展 库 定义 的 ， 
只 有 在 加 载 了 这 些 扩展 库 时 才 会 出 现 ， 要 么 动态 加 载 后 ， 要 么 在 编译 时 已 经 包括 进去 了 。PHP 中 预 
定义 了 许多 常量 ， 可 以 帮助 用 户 很 好 地 了 解 系统 的 当前 情况 。 以 下 就 是 几 个 PHP 预定 义 的 常量 ， 如 
表 2-15 所 示 。 


表 2-15 常用 常量 

常 量 名 说 有明 
fne 当前 PHP 程序 文件 名 
_line 当前 执行 语句 在 PHP 程序 文件 中 的 行 数 
PHP_version 当前 PHP 的 版 本 号 ， 注 意 必 须 大 写 
PHP os 当前 所 用 操作 系统 类 型 
E ERROR 指明 最 近 一 次 产生 不 可 恢复 的 错误 ， 注 意 必须 大 写 
E_WARING 指出 有 错误 ， 但 程序 可 以 继续 ， 注 意 必 须 大 写 
E PARSE 语法 错误 ， 分 析 器 将 被 停止 分 析 ， 注 意 必须 大 写 
E NOTICE 产生 异常 ， 但 不 一 定 是 错误 。 程 序 可 以 继续 ， 注 意 必须 大 写 


@ _ file 和 _line 是 双 下 划 线 ， 不 是 单 下 划 线 ,以 E 开头 的 常量 一 般 与 error_reporting 函数 联 用 ， 用 
提示 | 。 来 产生 相关 的 调试 出 错 信息 。 


2.5 ”表达 式 


表达 式 是 PHP 语言 的 基础 。 在 PHP 程序 中 ， 任 何 一 个 可 以 返回 值 的 语句 ， 都 可 以 看 作 表 达 式 。 
也 就 是 说 ， 表 示 式 是 一 个 短语 ， 能 够 执行 一 个 动作 ,并 具有 返回 值 。 一 个 表达 式 通常 由 两 部 分 构成 ， 
一 部 分 是 操作 数 ， 一 部 分 是 操作 符 。 


2.5.1 ”操作 数 


操作 数 就 是 在 进行 表达 式 计算 时 ， 需 要 使 用 的 数值 。 最 基本 的 表达 式 形式 是 常量 和 变量 。 当 输 
入 “$a=5”， 即 将 值 “5” 分 配给 变量 $a。“5”， 很 明显 ， 其 值 为 5， 换 句 话说 ,“5” 是 一 个 值 为 5 的 
表达 式 〈 在 这 里 ,“5” 是 一 个 整 型 常量 )。 赋 值 之 后 ， 所 期 待 情况 是 $a 的 值 为 5， 如 果 写 下 $b=$a， 
期 望 的 是 $b 犹如 $a=5 一 样 。 换 句 话说 ，$a 是 一 个 值 也 为 5 的 表达 式 。 如 果 一 切 运行 正确 ， 那 这 正 
是 将 要 发 生 的 正确 结果 。 在 这 里 ,“5” 和 $a 就 是 一 个 操作 数 。 稍 微 复杂 的 表达 式 例 子 就 是 函数 。 通 
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常 函数 不 会 仅仅 返回 一 个 静态 值 ， 而 可 能 会 计算 一 个 表达 式 。 当然, PHP 中 的 值 常常 并 非 是 整 型 的 。 
操作 数 的 使 用 示例 如 下 所 示 : 


S$at+? 


$value=$valuelt$value2; 


在 上 述 代 码 中 ，a、valuel、value2 分 别 都 是 操作 数 。 
2.5.2 ”操作 符 


操作 符 就 是 表达 式 要 执行 的 操作 的 类 型 ， 如 “+” 操 作 符 ， 表 示 该 表达 式 执行 了 一 个 求 和 运算 。 
对 于 学 过 其 他 语言 的 读者 ， 操 作 符 应 该 不 会 陌生 了 ， 但 是 在 PHP 中 ,操作 符 两 侧 的 操作 数 会 自动 进 
行 类 型 转换 ， 这 在 其 他 的 编程 语言 中 并 不 多 见 。 常 见 的 操作 符 ， 有 算术 操作 符 、 赋 值 操作 符 、 比 较 
操作 符 、 罗 辑 操作 符 等 。 

1. 算术 操作 符 

在 程序 语言 中 , 求 取 两 个 数 的 和 、 差 , 通常 采用 算术 操作 符 PHP 语言 中 的 算术 操作 符 如 表 2-16 
所 示 。 


表 2-16 算术 操作 符 


| 说 阴 | 结果 | 示例 | 说 明 
| Sai | 乘法 | sa 与 5b 的 各 


Sa 除 以 Sb 的 商 
Sa 除 以 Sb 的 余数 


在 上 述 的 算术 操作 符 中 ， 除 号 〈“/”) 总 是 返回 浮 点 数 ， 即 使 两 个 运算 数 是 整数 (或 由 字符 串 转 
换 成 的 整数 ) 也 是 这 样 。 取 模 $a%S$b 在 $a 为 负 值 时 的 结果 也 是 负 值 。 

2. 赋值 操作 符 

所 谓 的 赋值 操作 符 ， 完 成 的 操作 就 是 将 一 个 数值 赋 给 一 个 变量 。 常 用 的 赋值 操作 符 如 表 2-17 
所 示 。 


表 2-17 赋值 操作 符 


$a 等 于 5 $a/=5 
$a 等 于 $a 加 上 5 $a=5 
$a 等 于 $a 乘 以 5 


在 表 2-17 中 ，$a=5 表示 将 数值 5 赋值 给 变量 ga,“=” 是 一 个 赋值 符号 ， 而 不 是 一 个 等 号 ， 初 
学 者 尤其 要 注意 。 下 面 的 4 个 赋值 操作 符 ， 都 是 二 元 操作 符 ， 其 中 $a+=5 表示 $a=$a+5， 其 他 情况 以 
此 类 推 。 

3， 比 较 操作 符 

比较 操作 符 顾 名 思 义 就 是 比较 两 个 数值 的 大 小 ， 比 较 完 成 之 后 ， 返 回 的 值 为 布尔 值 。 比 较 表 达 
式 通常 作为 控制 语句 的 判断 条 件 。 常 用 的 比较 操作 符 如 表 2-18 所 示 。 


$a 等 于 $a 除 以 5 
拼接 赋值 | ”$a 等 于 $a 拼接 5 


表 2-18 ”比较 操作 符 


示 例 说 ”了 明 结 果 

$a—$b 等 于 True， 如 果 $a 等 于 Sb 

$a 一 -yb 全 等 True， 如 果 $a 等 于 Sb， 并 且 它 们 的 类 型 也 相同 
$a!=$b 不 等 Tme， 如 果 $a 不 等 于 $b 

$a<>$b 不 等 Tme， 如 果 $a 不 等 于 $b 

$a! 一 Sb 非 全 等 Tme， 如 果 $a 不 等 于 $b， 或 者 它们 的 类 型 不 同 
$a<$b 小 于 Trme， 如 果 $a 严格 小 于 $b 

$a>$b 大手 Tme， 如 果 $a 严格 大 于 $b 

$a<—$b 小 于 等 于 Tme， 如 果 $a 小 于 或 者 等 于 $b 

$a>=$b 大 于 等 于 Tme， 如 果 $a 大 于 或 者 等 于 $b 


在 上 述 比 较 操 作 符 中 ， 如 果 比 较 一 个 整数 和 字符 串 ， 则 字符 串 会 被 转换 为 整数 。 如 果 比 较 两 个 
数字 字符 串 ， 则 作为 整数 比较 。 除 了 上 面 的 比较 操作 符 外 ， 还 有 一 个 比较 特殊 的 比较 操作 符 ， 即 三 
元 操作 符 ， 该 运算 符 的 语法 格式 为 :“( 表 达 式 ) ? 值 1: 值 2”。 如 果 表 达 式 的 值 为 Tue， 则 整个 表 
达 式 返回 值 为 值 1， 如 果 表 达 式 值 为 False， 则 整个 表达 式 返 回 值 为 值 2。 其 使 用 示例 如 下 所 示 


<?php 
Ba = 间 
Sb = 27 
$c=($a>$b) ?$a: $b; 
echo($c); 

2 


执行 上 述 代码 ， 返 回 结果 为 2， 因 为 $a>$b 的 值 False， 故 此 时 应 返回 $b 的 值 。 

4， 逻辑 操作 符 

判断 某 一 个 对 象 是 否 符合 标准 ， 可 能 需要 判断 多 个 条 件 ， 这 时 就 要 用 到 逻辑 操作 符 。 逻 辑 操作 
符 通常 用 在 控制 语句 中 ， 如 寺 或 while 等 。 逻 辑 操作 符 使 用 的 操作 数 是 布尔 类 型 的 操作 数 。 常 用 的 
逻辑 操作 符 如 表 2-19 所 示 。 


表 2-19 ” 膛 辑 操作 符 
示例 | 说 阴 | 结 果 
$a and $b and (逻辑 与 ) True， 如 果 $a 与 $b 都 为 True 
$a or $b or (逻辑 或 ) True， 如 果 $a 或 $b 任 一 为 Tue 
$a xor $b xor (逻辑 异 或 ) True， 如 果 $a 或 $b 任 一 为 Tme， 但 不 同时 为 True 
1$a not〈 逻 辑 非 ) True， 如 果 $a 不 为 True 
$a &e& $b True， 如 果 $a 与 $b 都 为 True 


$all $b or (有 逻辑 或 ) True， 如 果 $a 或 $b 任 一 为 Tme 


在 上 述 的 逻辑 操作 符 中 ,“ 逻 辑 与 ”和 “逻辑 或 ”有 两 种 不 同形 式 的 运算 符 ， 原 因 是 它们 运算 
的 优先 级 不 同 。 其 使 用 示例 如 下 所 示 : 
<?php 
$value=67; 
if($value>60 and S$value<80) 
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echo "该 学 生成 绩 良 好 "; 
if($value>=80 and $value<=100) 
echo "该 学 生成 绩优 秀 "; 


2> 
在 上 述 代码 中 , 使 用 了 风 辑 操作 符 and 判断 两 个 条 件 是 否 都 是 True 值 ， 如 果 都 为 Tme， 则 输出 
相应 的 语句 。 
5. 字符 串 操作 符 
D 


常用 的 字符 串 操 作 符 有 两 个 ， 一 个 是 连接 运算 符 (“.”)， 它 返回 其 左右 参数 连接 后 的 字符 串 。 
另 一 个 是 连接 赋值 运算 符 (“.=”)， 它 将 右边 参数 附加 到 左边 的 参数 后 面 。 如 表 2-20 所 示 。 


表 2-20 字符 串 操作 符 


示 例 结 果 
$a="abc"."def" $a 赋值 为 字符 串 “abcdef” 
$a . ="ghijkl" $a 等 于 它 的 当前 值 与 “ghijkl” 的 拼接 结果 


字符 串 的 处 理 功 能 还 有 很 多 ， 后 面 的 章节 将 会 陆续 提 到 。 

6. 自 增 和 自 减 操作 符 

自 增 和 自 减 操作 符 是 一 元 操作 符 ， 可 以 使 某 一 个 变量 自动 加 1 或 者 减 1。 常 用 的 自 增 和 自 减 操 
作 符 如 表 2-21 所 示 。 
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表 2-21 自 增 和 自 减 操作 符 


| 说 明 | 结果 | 示例 | 说 明 | 结果 
| 前 加 | $a 的 值 加 1， 然后 返回 S|】 --sa。 | 前 减 | $a 的 值 减 1， 然 后 返回 $a 
返回 Sa， 然后 将 $a 的 值 加 1 | sa-- | 后 减 ”| 返回 $a, 然后 将 $a 的 值 减 1 


从 表 2-21 中 可 以 看 出 , 操作 符 放置 的 位 置 不 同 , 会 引起 不 同 的 执行 结果 。 其 使 用 示例 如 下 所 示 : 
<?php 


echo "<h3>Postincrement</h3>"; 

Sa = 57 

echo "Should be SEE Sa P<br AN 
echo “should be 6:. "$a "br />\nns 
echo "<h3>Preincrement</h3>"; 

$a= 5; 

echo "Should be 6: " . ++$a . "<br />\n"; 
echo Should be 6 " . Sa = "<br />\n"s 
echo "<h3>Postdecrement</h3>"; 

Sa = 57 

echo "Shoula be 5: ”- $a== . "<br />\n"; 
echo "Should be 4: ™ » $a « "<br />\n"s 
echo "<h3>Predecrement</h3>"; 


Sa = 5; 

echo “Should pe 4°" -$a a “<bre >NDCA 
echo “Should be 4: " . $a . “<br />\n"; 
?> 


7. 错误 控制 操作 符 
PHP 支持 一 个 错误 控制 运算 符 : @。 将 其 放置 在 一 个 PHP 表达 式 之 前 ,该 表达 式 可 能 产生 的 任 
何 报错 消息 都 会 被 忽略 掉 。 错 误 控制 操作 符 的 使 用 示例 如 下 所 示 : 


<?php 

$my_file = efile ('non existent file') or 
die ("Failed opening file: error was '$php errormsg'"); 

$value = @$cache[$key]; 

2 


@ 运 算 符 只 对 表达 式 有 效 。 对 新 手 来 说 一 个 简单 的 规则 就 是 : 如 果 能 从 某 处 得 到 值 ， 就 能 在 它 


前 面 加 上 @ 运 算 符 。 例 如 ， 可 以 把 它 放 在 变量 、 函 数 或 include0 函 数 调用 、 常 量 等 之 前 。 不 能 把 它 第 
放 在 函数 或 类 的 定义 之 前 ， 也 不 能 用 于 条 件 结构 ， 如 让 和 foreach 等 。 加 
8. 位 运算 符 
位 运算 符 允 许 对 整 型 数 中 指定 的 位 进行 置 位 。 如 果 左 右 参 数 都 是 字符 串 ， 则 位 运算 符 将 操作 字 时 
符 的 ASCII 值 。 其 详细 信息 如 表 2-22 所 示 。 基 
表 2-22 ”位 运算 符 相 
示例 结果 
Sat $b 将 $a 和 Sb 中 都 为 1 的 位 设 为 1 
$alsb 将 $a 或 者 $b 中 为 1 的 位 设 为 1 
$a^ $b 将 $a 和 Sb 中 不 同 的 位 设 为 1 
一 3a 将 $a 中 为 0 的 位 设 为 1， 反之 亦 然 
$a << $b 将 $a 中 的 位 向 左 移动 Sb 次 (每 一 次 移动 都 表示 “ 乘 以 2”) 
$a>> $b 将 $a 中 的 位 向 右 移动 $b 次 (每 一 次 移动 都 表示 “ 除 以 2”) 
位 运算 符 的 使 用 示例 如 下 所 示 : 
<?php 


echo 12 ^ 9; // 输出 为 “5” 
echo "12"” ^"9"; // 输出 退 格 字 符 (ASCII 8) 
A ML NASCIE A 2 (9% MASCII SII = #0 
echo "hallo" ^ "hello"; // 输出 ASCII 值 #0 #4 #0 #0 #0 
J Mae ne = 


?> 


在 PHP 中 ， 如 果 程序 涉及 到 一 个 复杂 的 表达 式 ， 这 个 表达 式 中 包含 了 多 种 类 型 的 运算 符 ， 则 需 
要 考虑 表达 式 的 优先 级 。 操 作 符 的 优先 级 是 指 表达 式 的 执行 顺序 ， 其 详细 信息 如 表 2-23 所 示 。 


表 2-23 运算 符 优先 级 


new 
amay0 
递增 /递减 运算 符 


卫 
ua 
3 
十 
< 
op 
[3 
| x 
完 
全 
学 
习 
手 
册 


结合 方向 运 算 符 附加 信息 
非 结合 !~— (int) (floab (string) (array) (object) @ 类 型 
左 */% 算数 运算 符 
左 十 一 算数 运算 符 和 字符 串 运算 符 
左 <<>> 位 运算 符 
非 结合 < 二 > 一 比较 运算 符 
非 结合 一 二 -一 :一 比较 运算 符 
左 & 位 运算 符 和 引用 
左 ^ 位 运算 符 
左 | 位 运算 符 
左 && 逻辑 运算 符 
左 1 逻辑 运算 符 
左 三 元 运算 符 
右 ==-= 村 上 -=%W=&=| 伟 <<=>~ 赋值 运算 符 
左 and 逻辑 运算 符 
左 XOr 逻辑 运算 符 
左 or 逻辑 运算 符 
左 多 处 用 到 


2.6 ”控制 结构 


任何 PHP 脚本 程序 都 是 由 一 系列 语句 构成 的 。 语 句 可 以 是 赋值 语句 、 函 数 调 用 、 循 环 语句 、 条 
件 语句 ， 甚 至 是 空 语句 。 语 句 通常 以 分 号 结束 。 此 外 ， 还 可 以 用 大 括号 将 一 组 语句 封装 成 一 个 语句 
组 ， 语 句 组 本 身 可 以 看 作 一 行 语句 ， 执 行 一 定 的 功能 。 本 节 将 介绍 各 种 语句 类 型 。 


2.6.1 ”条件 语句 


条 件 语 句 在 PHP 中 非常 普遍 ， 它 是 PHP 程序 的 主要 控制 流程 之 一 。 通 常情 况 下 ， 在 客户 端 获 
得 一 个 参数 ， 根 据 传 入 的 参数 值 ， 做 出 不 同 的 响应 。 在 这 个 条 件 的 判断 过 程 中 ， 符 合 条 件 的 代码 执 
行 了 ， 其 他 的 代码 却 没 有 执行 。PHP 技术 中 ， 条 件 语句 有 多 种 类 型 ， 分 别 为 ff 语 句 、if-else 语句 、 
if-elseif-else 语句 、switch 语句 等 。 其 详细 信息 如 表 2-24 所 示 。 


表 2-24 条 件 语句 


语 多 
站 语句 


语法 格式 


证 (expD{ 
statement 


| 


语句 功能 
expr 是 一 个 表达 式 ， 结 果 值 为 布 
尔 值 。 如 果 布 尔 值 为 Te， 就 执 


行 下 面 的 大 括号 中 的 statement 
语句 , 如 果 值 为 False, 将 会 忽略 


语句 示例 
<?php 
$a=4: 
$b=8; 
if($a<$b){ 


语 名 语法 格式 语句 功能 语句 示例 
statement 语句 。 如 果 statement | echo ("变量 a 的 值 小 于 b 的 值 "); 
语句 只 有 一 条 ， 这 时 大 括号 可 以 | $a=$b: 
省 略 
?> 
让 else 语句 | if (expD){ 使 用 让 语句 只 能 对 一 种 情况 做 出 
statementl 反应 ， 对 于 该 情况 之 外 的 其 他 情 $a=4: 
况 无 法 做 出 相应 的 响应 。 如 果 在 $b=9: 
deet 程序 中 ， 要 求 对 不 符合 条 件 的 情 ($a> $b) 
statement2 况 也 做 出 响应 ， 这 时 需要 使 用 echo "变量 a 大 于 变量 b"; 
} 让 else 语句 ce 
echo "变量 a 小 于 等 于 变量 b"; 
if-elseif-else | if (expr){ if-else 语句 只 能 判断 两 种 情况 ， <?php 
语句 statementl | 要 么 符合 条 件 ， 要 么 不 符合 条 Svalue=78; 
} 件 。 如果 要 求 对 每 一 种 情况 都 做 if($value<60) 
elseif(expr2) 出 反应 ，ifelse 语句 就 不 能 完成 {echo(" 该 学 生成 绩 不 及 格 ");} 
{statement2) | 这 样 的 任务 。 这 时 可 以 使 用 elseif($value>=60 and $value<80) 
elseiftexpr3) if-elseif-else 语句 来 完成 {echo(" 该 学 生成 绩 良 好 "):} 
{statement3} elseif($value>=80 and $value<100) 
me {echo(" 该 学 生成 绩优 秀 "):} 
else{ else{echo(" 该 学 生成 绩 满分 ");} 
statement4 ?> 
} 
switch 语句 | <?php 多 分 支 语句 ， 用 来 判断 多 种 情况 <?php 


switch (expr) { 
case valuel: 
statementl: 
break: 
case value2: 
statement2: 
break: 
case Valuen: 
statementn: 
break: 
default: 


statement(n+1): 


} 


和 


的 执行 


$i=" 西 瓜 "; 
Switch ($1) { 

case "苹果 ": 

echo "这 是 一 个 苹果 "; 
break: 

case "香蕉 ": 

echo "这 是 一 个 香 燕 ": 
break: 

case "西瓜 ": 

echo "这 是 一 个 西瓜 "; 
break: 

default: 

echo "这 不 是 一 个 瓜 果 ": 
} 

?> 
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2.6.2 ”循环 语句 


循环 语句 主要 解决 代码 的 重复 执行 ， 也 是 程序 的 控制 流程 之 一 。PHP 中 采用 的 循环 语句 和 C、 
Java 语言 等 的 循环 语句 比较 相似 。 在 PHP 中 常用 的 循环 语句 有 下 面 几 种 形式 ， 分 别 为 while 循环 、 
do-while 循环 、for 循环 、foreach 循环 等 。 其 详细 信息 如 表 2-25 所 示 。 


表 2-25 ”循环 语句 
语 名 语法 格式 语句 功能 语句 示例 
while while (expr) 一 种 比较 简单 的 循环 。while 语句 的 含义 | <?php 
{ 很 简单 ， 它 告诉 PHP， 只 要 while 语句 中 | $i= 1: 
statement 表达 式 expr 的 值 为 TRUE， 就 重复 执行 | while ($i <= 10){ 
} 长 套 中 的 循环 语句 。 表 达 式 的 值 在 每 次 开 print $i: 
始 循环 时 检查 ,所 以 即使 这 个 值 在 循环 语 ER 
名 中 改变 了 , 语句 也 不 会 停止 执行 , 直到 
本 次 循环 结束 。 有 时 候 如 果 while 表达 式 
的 值 一 开始 就 是 FALSE， 则 循环 语句 一 
次 都 不 会 执行 
do-while | dof 是 while 循环 一 种 变种 ， 如 果 满 足 条 件 ， | <?php 
statement: 都 是 重复 的 执行 指定 的 代码 上 者 三 区 
}while(exp?): dof 
echo $i; 
} while ($i > 0):; 
和 
for 循环 for (exprl; expr2: expr3) | 一 个 比较 复杂 的 循环 机 制 , 其 功能 也 是 强 | <?php 
{ 大 的 。 第 一 个 表达 式 〈exprl) 在 循环 开 | for ($i= 1; $i <= 10; $it+) { 
statement 始 前 无 条 件 求 值 一 次 。expr2 在 每 次 循环 echo $i: 
} 开始 前 求 值 。 如 果 值 为 TRUE， 则 继续 循 | } 
环 ， 执 行 嵌 套 的 循环 语句 。 如 果 值 为 
FALSE， 则 终止 循环 。expr3 在 每 次 循环 
之 后 被 求 值 ( 执 行 )。 每 个 表达 式 都 可 以 
为 空 。expr2 为 空 意味 着 将 无 限 循环 下 去 
(和 C 一样 ，PHP 认为 其 值 为 TRUE) 
foreach foreach (array_expression | 一 种 主要 用 于 遍历 数组 的 循环 语句 ,该 语 | <?php 


语句 as $value) 句 从 PHP 4.0 版 本 开始 使 用 , 和 Perl 语言 | $arr = array(1. 2. 3. 4): 
{ 比较 相似 。 从 PHP 5.0 开始 ， 也 可 以 用 于 | foreach ($arr as $value) { 
Statement 对 象 $value = $value * 2: 


} echo($value): 
}/ 输 出 的 结果 为 2，4，6，8 


> 


do-while 和 while 循环 非常 相似 ， 区 别 在 于 表达 式 的 值 是 在 每 次 循环 结束 时 检查 而 不 是 在 开始 
时 检查 。 和 正规 的 while 循环 主要 的 区 别 是 do-while 循环 语句 保证 会 执行 一 次 (表达 式 的 真 值 在 每 
次 循环 结束 后 检查 )， 然 而 在 正规 的 while 循环 中 就 不 一 定 了 《表达 式 的 真 值 在 循环 开始 时 检查 ， 如 
果 一 开始 就 为 False 则 整个 循环 立即 中 止 )。 


2.6.3 break 和 continue 语句 


为 了 帮助 程序 员 更 加 精确 地 控制 整个 流程 ， 方 便 程序 的 设计 ，PHP 提供 了 两 种 跳 转 语句 : break 
和 continue。 其 详细 信息 如 表 2-26 所 示 。 


表 2-26 ” 跳 转 语句 
语 名 功能 描述 示 例 
break 语句 表示 程序 的 控制 权 强制 性 地 从 当前 的 语句 块 〈 主 要 为 循环 语 | <?php 

名) 跳 转 出 来 ， 执 行 语句 块 下 面 的 语句 。break 可 以 结束 当前 | $arr = array("red","green","blue"); 

for、foreach、while、do-while 或 者 switch 结构 的 执行 foreach ($arr as $value) { 
echo($value): 
if($value=="green") 

{break:} 


} 
echo(" 下 面 的 语句 "); 


?> 


continue 语句 | 表示 退出 当前 的 循环 而 执行 循环 语句 的 下 一 次 循环 。 该 语句 | <?php 
可 以 用 在 循环 语句 和 switch 语句 中 。 和 break 语句 的 区 别 是 for($Si=0:$i<5S:++$i { 
后 者 跳出 了 整个 循环 ， 前 者 只 跳出 当前 循环 ifGi 一 2) 
{ 

continue; 

print "Sin"; 

有 

echo($i: 


2.6.4 文件 包含 语句 


在 一 个 PHP 页 面 中 封装 一 段 执行 特定 功能 的 代码 ， 如 显示 指定 的 样式 、 当 前 的 时 间 等 。 然 后 在 
另外 一 个 页 面包 含 该 页 面 ， 这 样 做 的 好 处 是 ， 节 省 了 编码 的 时 间 ， 达 到 了 功能 的 相互 分 离 。 在 一 个 
PHP 页 面 中 包含 另外 一 个 页 面 ， 有 4 个 语句 可 以 实现 ， 分 别 为 require、include、require_once、 
include_ once 等 。 其 详细 信息 如 表 2-27 所 示 。 


表 2-27 包含 语句 
语 多 功能 描述 示 例 
includeO 包含 并 运行 指定 文件 见 例 2-6 


包含 并 运行 指定 文件 。require0 和 include0 除 了 处 理 失败 之 外 ， 在 各 方 | <?php 

面 都 完全 一 样 。include0 产 生 一 个 警告 ， 而 require0 则 导致 一 个 致命 错 | require 'prepend php': 
误 。 换 句 话 说 , 如 果 想 在 丢失 文件 时 停止 处 理 页 面 , 应 该 使 用 requireO0。| require (somefile txt: 
include0 就 不 是 这 样 ， 脚 本 会 继续 运行 。 同 时 也 要 确认 设置 了 合适 的 | ?> 


IequireO 


include path 
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语 和 句 功能 描述 
include_once0 | 在 脚本 执行 期 间 包 含 并 运行 指定 文件 。 此 行为 和 include0 语句 类 似 ， 
唯一 的 区 别 是 如 果 该 文件 中 的 代码 已 经 被 包含 了 ， 则 不 会 再 次 包含 。 
如 同 此 语句 名 字 暗 示 的 那样 ， 只 会 包含 一 次 。include_once0 应 该 用 于 
在 脚本 执行 期 间 同一 个 文件 有 可 能 被 包含 超过 一 次 的 情况 下 ， 想 确保 
它 只 被 包含 一 次 以 避免 函数 重 定义 、 变 量 重新 赋值 等 问题 
require_once() | 确保 文件 只 能 被 包含 一 次 ， 在 require_once0 后 面 ， 试 图 包含 相同 的 文 
件 将 会 被 忽略 。 该 语句 的 使 用 同 include_once0 的 使 用 基本 一 样 


<php 
include_ once '1.txt; 
include_once(1.txt): 


?> 


<?php 
require_once("a.php"); 
require_once("A.php"); 
?> 


当 一 个 文件 被 包含 时 ， 其 中 所 包含 的 代码 继承 了 include 所 在 行 的 变量 范围 。 从 该 处 开始 , 调用 
文件 在 该 行 处 可 用 的 任何 变量 在 被 调用 的 文件 中 也 都 可 用 。 但 是 所 有 在 包含 文件 中 定义 的 函数 和 类 
都 具有 全 局 作用 域 。 寻 找 包 含 文件 的 顺序 先是 在 当前 工作 目录 的 相对 include_ path 下 寻找 ， 然 后 是 
在 当前 运行 脚本 所 在 目录 相对 的 include_path 下 寻找 。 若 文件 中 有 一 句 include"b.php"， 则 寻找 bphp 
的 顺序 先是 /www/， 然 后 是 /www/include/。 如 果 文 件 名 以 “./” 或 者 “../” 开 始 ， 则 只 在 当前 工作 目 
录 相 对 的 include_path 下 寻找 。 

下 面 创建 一 个 案例 ， 演 示 在 PHP 页 面 中 包含 另外 一 个 页 面 。 代 码 如 下 所 示 : 

案例 2-6 

<?php 

echo" 下 面 是 要 包含 的 页 面 "; 


echo "<hr>"; 


include "test.php"; 
echo("<br>"); 
include "1.txt"; 

?> 

<hr> 

<?php 

echo "包含 成 功 "; 


人 


将 该 文件 保存 为 include.php, 并 放 到 指定 的 位 
置 .打开 正 浏 览 器 ,在 地 址 栏 中 输入 http://localhost: v 
8080/mmyweb/include.php， 单 击 【 转 到 】 按 钮 ， 会 显 | 下 再 要 包 的 再 
示 如 图 2-6 所 示 的 窗口 。 在 本 案例 中 , 使 用 include 其 和 2 
语句 包含 了 一 个 PHP 页 面 和 一 个 txt 文件 。 

在 这 里 需要 注意 的 是 , Windows 系统 下 不 区 分 
大 小 写 。 


图 2-6 包含 页 面 


>》 include0 包 含 文件 中 的 所 有 代码 都 会 继承 其 调用 者 位 置 处 的 变量 作用 域 . 只 有 启用 allow_url fopen( 这 
提示 是 默认 值 )， 才 可 以 在 TequireO 中 使 用 URL. 


/ 学 习 目标 |oujecuve 


无 论 任何 一 种 编程 语言 ， 都 会 出 现 功能 重复 的 编码 ， 如 连接 数据 库 、 显 示 当 前 时 间 等 。 对 于 大 
规模 的 程序 ， 这 样 的 代码 出 现 的 次 数 更 多 ， 如 果 整 个 程序 出 现 这 样 的 情况 ， 对 于 程序 的 修改 和 维护 
将 是 一 个 很 大 的 灾难 。 对 于 这 些 功能 相同 的 代码 ,在 PHP 中 是 以 函数 的 形式 进行 封装 的 。 程 序 中 其 
他 地 方 需要 此 类 功能 ， 直 接 调 用 函数 就 可 以 了 。 如 果 程 序 出 现 问题 ， 只 需 修 改 这 些 函 数 即 可 。 

本 章 将 学 习 PHP 函数 的 知识 ， 包 括 如 何 创建 和 调用 函数 、 传 递 输入 、 为 调用 者 返回 一 个 或 多 个 
值 ， 以 及 创建 和 包含 函数 库 。 此 外 ， 还 将 了 解 递归 〈recursive) 和 变量 〈variable) 函数 的 使 用 。 


FY 内 容 摘要 1apstsct 


掌握 PHP 中 函数 的 调用 

熟练 掌握 在 PHP 中 自 定义 函数 
掌握 PHP 函数 参数 传 值 的 4 种 方式 
掌握 PHP 函数 返回 值 的 使 用 

掌握 PHP 嵌 套 和 递归 函数 的 使 用 
掌握 PHP 变量 函数 的 使 用 

掌握 PHP 数学 和 日 期 函数 

熟练 掌握 在 PHP 中 自 定义 函数 库 


3.1 调用 函数 


在 PHP 中 有 两 类 函数 , 分 别 为 系统 预定 义 函数 和 用 户 自 定义 函数 。 系 统 预 定义 函数 也 称 内 置 函 
数 ， 标 准 的 PHP 发 行 包 中 有 1000 多 个 标准 函数 ， 其 中 很 多 都 会 在 本 书 中 出 现 。 假 设 函数 库 已 经 编 
译 到 安装 发 行 包 中 ， 或 者 通过 include0 或 require0 语 句 包含 了 相应 的 函数 库 ， 那 么 通过 指定 函数 名 
就 可 以 调用 函数 。 同 样 用 户 的 自 定义 函数 也 是 通过 指定 函数 名 来 调用 。 对 于 用 户 自 定义 函数 ， 将 在 
下 一 节 详 细 地 进行 说 明 。 本 节 将 以 系统 预定 义 函数 为 例 ， 对 函数 的 调用 进行 说 明 。 

在 数学 函数 的 类 别 中 ， 有 abs0 和 rand0 两 个 函数 ， 分 别 是 求 数 的 绝对 值 和 产生 一 个 随机 数 。 下 
面 创建 一 个 案例 ， 演 示 调 用 这 两 个 函数 。 该 案例 的 代码 如 下 所 示 : 


案例 3-1 

<?2php 
echo abs(-4); 
echo "<hr>"; 


echo "下 面 是 产生 的 随机 数 "; 
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for ($i=0;$i<7;$i++) { 
$value=rand (); 
echo ("<br>"); 
echo ( (int) $value); 


. 


> 


将 上 述 代码 保存 ， 文 件 名 为 Maphp， 保 存 位 “ER 
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置 在 C:\Web\apache\htdocs\myweb 目录 下 。 打开 正 
浏览 器 ， 在 地 址 栏 中 输入 http://localhost:8080/ 
myweb/Ma.php， 单 击 【 转 到 】 按 钮 ， 会 显示 如 图 
3-1 所 示 的 窗口 。 

从 上 面 的 案例 可 以 看 出 , 函数 的 调用 直接 使 用 
函数 名 就 可 以 了 。 如 果 该 函数 需要 参数 , 就 直接 赋 
予 参数 ,应 注意 参数 的 类 型 .但 是 这 里 有 一 个 前 提 ， 
就 是 保证 该 函数 库 被 包含 了 进来 。 图 3-1 函数 的 调用 


3.2 ”用 户 自 定义 函数 


PHP 提供 了 大 量 的 系统 预定 义 函数 ， 加 强 了 PHP 的 功能 ， 并 提高 了 程序 员 的 开发 速度 。 但 是 ， 
系统 预定 义 函数 只 适用 于 一 般 的 情况 , 对 于 新 的 环境 或 者 程序 特殊 的 要 求 , 预定 义 函数 也 无 能 为 力 ， 
这 时 可 以 自己 编写 函数 解决 问题 。 本 节 将 对 用 户 自 定义 函数 的 创建 、 使 用 进行 详细 的 介绍 。 


3.2.1 创建 函数 


创建 一 个 用 户 自 定义 函数 ， 可 以 方便 程序 的 开发 并 加 快 开发 速度 。 如 果 程 序 出 现 了 变动 ， 可 以 
快速 地 调整 自 定义 函数 ， 避 免 了 多 处 修改 代码 。PHP 中 的 函数 (function》 和 C 语言 中 的 一 样 ， 包 
括 有 返回 值 的 函数 及 无 返回 值 的 函数 ， 不 像 Pascal 分 成 函数 〈function) 和 过 程 (procedure) 那么 复 
杂 。 在 PHP 中 ， 创 建 一 个 函数 的 语法 格式 如 下 所 示 : 


function Name ($argl, $arg2, $arg3...S$argn) // 函 数 名 和 参数 名 
{ 

. . .// 函 数 体 ， 即 需要 执行 的 语句 

return $Name;// 将 某 个 变量 值 返回 到 程序 中 
| 


在 上 述 代码 中 , Name 表示 函数 的 名 称 , $argl1、$arg2 等 表示 函数 的 参数 , 大 括号 中 间 为 函数 体 ， 
即 函数 的 执行 语句 。retum 表示 该 函数 具有 返回 值 ，retum 语句 是 可 选 的。 在 函数 的 名 称 上 ，PHP 对 
于 大 小 写 的 限制 很 松散 ， 可 以 在 定义 函数 时 写成 大 写 形式 的 函数 名 ， 而 在 使 用 时 用 小 写 形式 的 函数 
名 。 总 之 ， 对 函数 而 言 ， 不 区 分 大 小 写 ， 只 要 注意 名 称 没有 重复 就 行 了 。 有 效 的 函数 名 以 字母 或 下 
划 线 开始 ， 后 面 跟 字母 、 数 字 或 下 划 线 。 用 户 的 信息 可 以 通过 函数 的 参数 来 进行 传递 。 如 果 使 用 多 


个 参数 ， 需 要 用 逗号 隔 开 。PHP 支持 的 参数 传递 方式 有 : 通过 值 传递 (passing by value)、 通 过 引 上 
传递 (passing by reference)、 默 认 方式 和 可 变 参 数列 表 。 任 何 有 效 的 PHP 代码 都 有 可 能 出 现在 函数 
下 面 创建 一 个 案例 ， 演 示 创 建 自 定义 函数 的 过 程 。 该 案例 的 代码 如 下 所 示 : 


案例 3-2 
<?php 
function Da(){ 
echo Date('Y-m-j'); 
} 
function Sum($a, $b){ 
$c=$at+$b; 
return $c; 
} 
echo (" 现 在 时 间 为 :") 7 
da () 7 
echo ('<br>4 和 7 两 个 数 的 和 为 ') 7 


Ssum=sum(4,7) 7 


echo ($sum); 
?> 


将 上 述 代码 保存 ， 文 件 名 为 Funphp， 并 保存 到 指定 的 位 置 。 打 开 正 浏览 器 ， 在 地 址 栏 中 输入 
http://localhost:8080/myweb/Fun.pbp， 单 击 【 转 到 】 
按钮 ， 会 显示 如 图 3-2 所 示 的 窗口 。 ee 人 
在 本 案例 中 ， 创 建 两 个 自 定 义 函数 ， 一 个 是 时 seen 
Da0， 用 来 显示 当前 日 期 ， 一 个 是 sumO， 用 来 求 | We 和 i 
取 两 个 数 的 和 。Da0 函 数 只 是 完成 一 种 功能 , 没有 
返回 值 。sumO 函 数 具 有 返回 值 ， 返 回 两 个 数 的 和 。 
PHP 中 的 所 有 函数 和 类 都 具有 全 局 作用 域 ， 可 以 
在 内 部 定义 外 部 调用 ， 反 之 亦 然 。PHP 不 支持 函数 重 载 ， 也 不 可 能 取消 定义 或 者 重 定义 已 声明 的 函 
数 。 通 常 把 在 创建 函数 时 ， 命 名 的 参数 称 为 形 参 ， 如 本 案例 中 a、b。 在 函数 调用 时 ， 给 函数 传递 的 
参数 称 为 实 参 ， 如 本 案例 中 4、7 等 。 


图 3-2 创建 函数 


0 f 函数 名 不 区 分 大 小 写 ， 但 是 在 调用 函数 时 ， 通 常 使 用 其 在 定义 时 形式 。 
示 


3.2.2 ” 按 值 传递 参数 


PHP 默认 的 参数 传递 方式 是 通过 值 传递 (passing by value)。 这 种 传递 方式 仅仅 把 函数 外 部 变量 
的 值 备 份 一 个 副 件 ， 然 后 赋 给 函数 内 部 的 局 部 变量 。 在 函数 处 理 完成 后 ， 该 外 部 变量 的 值 不 发 生 改 
变 ， 除 非 在 函数 内 部 声明 了 该 外 部 变量 ， 并 做 了 改动 ， 即 每 次 调用 外 部 变量 时 ， 外 部 变量 会 形成 一 


第 
3 
章 
数 


个 备份 数值 ， 把 备份 数值 传递 给 函数 ， 修 改 的 结果 只 影响 备份 数值 ， 而 原来 的 值 没有 改变 。 
下 面 创建 一 个 案例 ， 演 示 函 数 按 值 传递 参数 。 该 案例 的 代码 如 下 所 示 : 


案例 3-3 
<?php 
// 第 一 个 函数 不 影响 外 部 变量 的 值 
// 第 二 个 函数 影响 外 部 变量 的 值 
$string ="hello"™; 
四 function hi($str) 
{ 
$str.="all"; 
echo $str; 
hi($string); 
Soho Do 
echo $string; 
function hill($string){ 
global $string; 
$str="all"; 
$string.=$str; 
echo $str; 
} 
echo "<hr>"; 
hil($string) 7 
echo "<hr>"; 
echo $string; 
2 


将 上 述 代码 保存 ， 文 件 名 为 argl.php， 并 保存 到 指定 的 位 置 。 打 开 正 浏览 器 ， 在 地 址 栏 中 输入 
http://localhost:8080/myweb/arg1.php， 单 击 【 转 到 】 按 钮 ， 会 显示 如 图 3-3 所 示 的 窗口 。 
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图 3-3 按 值 传递 


在 本 案例 中 ， 共 创建 了 两 个 函数 ， 分 别 为 hi0 和 hi10， 这 两 个 函数 都 使 用 了 一 个 共同 的 外 部 变 
量 string。 在 函数 hi0 中 ， 在 传递 过 来 的 参数 str 的 基础 上 ， 把 str 和 字符 串 “all” 连 接 起 来 ， 然 后 输 
出 。 在 下 面 调用 函数 时 ， 给 函数 hi ($string) 传递 了 一 个 变量 string， 这 时 string 变量 只 是 将 变量 副 
本 传递 过 去 ， 没 有 影响 $string 的 值 ， 故 在 下 面 调用 函数 时 输出 helloall， 而 string 的 值 还 是 hello。 在 
第 二 个 函数 hi10 中 ， 为 了 能 够 改变 外 部 变量 的 值 ， 首 先 声 明 该 变量 为 全 局 变量 ， 再 使 用 该 变量 就 是 
变量 的 数值 本 身 ， 而 不 是 变量 的 副本 了 ， 在 函数 中 语句 “string.=str” 表 示 ， 将 传递 字符 串 string 和 
str 连接 起 来 ， 这 时 变量 string 的 改变 影响 了 原来 的 数值 ， 所 以 在 输出 时 ，string 的 值 就 变 成 helloall。 


3.2.3 ” 按 引用 传递 参数 


在 函数 的 参数 传递 中 ， 另 外 一 种 方式 是 引用 《〈 地 址 ) 传递 方式 。 在 这 种 方式 下 ， 实 参 的 内 存 地 
址 被 传递 到 形 参 中 ， 在 函数 内 部 对 形 参 的 任何 修改 都 会 影响 到 实 参 ， 因 为 它们 被 存储 到 同一 个 内 存 
地 址 。 函 数 返回 后 ， 实 参 的 值 将 会 发 生变 化 。 引 用 传递 参数 的 形 参 和 实 参 都 是 针对 同一 个 块 地 址 修 
改 的 。 

下 面 创建 一 个 案例 ， 演 示 函 数 按 引 用 传递 参数 。 该 案例 的 代码 如 下 所 示 : 


案例 3-4 
<?php 
function str(g$string){ 
$string.=" 第 2 个 字符 串 "7 

3 
$str=" 第 1 个 字符 串 "; 
Sotto 
echo $str; 
echo ("<hr>™ 

二 

<?php 

function strl($string){ 
$string.=" 第 2 个 字符 串 "; 

} 
$str=" 第 1 个 字符 串 "; 
strl($str); 
echo $str; 
echo("<hr>"); 
strl(g&$str); 
echo $str; 

?> 


将 上 述 代码 保存 ， 文 件 名 为 arg2.php， 并 保存 到 指定 的 位 置 。 打 开 正 浏览 器 ， 在 地 址 栏 中 输入 
http://localhost:8080/myweb/arg2.php， 单 击 【 转 到 】 
按钮 ， 会 显示 如 图 3-4 所 示 的 窗口 。 Ee 

在 本 案例 中 ， 创 建 了 两 个 函数 st0 和 str10， 痢 see Ee 
是 实现 按 引 用 传递 参数 ， 只 是 方式 不 同 罢了 。 在 第 一 | 和 13a 
个 函数 中 ,将 传递 过 来 的 参数 值 string 和 现 有 的 字符 | 和 
串 进行 连接 ， 该 函数 的 参数 的 书写 形式 为 &$string， 
表示 在 参数 的 传递 中 使 用 了 按 地 址 传递 的 方式 。 在 下 
面 调 用 时 ， 直 接 把 变量 的 地 址 传递 到 函数 中 ,函数 中 
的 任何 修改 都 会 作用 到 变量 的 数值 。 在 第 二 个 函数 中 ， 完 成 的 功能 同样 是 将 参数 string 和 指定 的 字 
符 串 连接 起 来 ， 其 书写 形式 没有 什么 变化 ， 需 要 注意 的 是 ， 在 下 面 调用 这 个 函数 时 ， 可 以 使 用 语句 
str(&$str); 设 置 参 数 的 传 值 是 按 引 用 传递 。 可 以 得 出 ， 如 果 希 望 一 个 函数 参数 通过 引用 被 传递 ， 可 以 
预先 的 函数 定义 中 的 参数 名 前 加 符号 “& ”。 


3-4” 按 引用 传递 参数 
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3.2.4 ”默认 参数 值 


除了 上 面 所 有 的 传递 参数 的 方法 外 ， 还 可 以 使 用 预定 义 的 默认 参数 。 在 未 指定 参数 的 情况 下 ， 
函数 使 用 默认 值 作为 函数 的 参数 ， 在 提供 了 参数 的 情况 下 ， 函 数 使 用 指定 的 参数 。 
下 面 创 建 一 个 案例 ， 演 示 使 用 默认 参数 值 。 该 案例 的 代码 如 下 所 示 : 


案例 3-5 
<?php 
function cook($str=" 早 上 好 ") { 
return "大 家 $str"; 
} 
echo cook(); 
echo. “<hr>"y 
echo cook ("请 坐 F"); 
2 


将 上 述 代 码 保存 ， 文 件 名 为 arg3.php， 并 保存 到 指定 的 位 置 。 打 开 正 浏览 器 ， 在 地 址 栏 中 输入 
http://localhost:8080/myweb/arg3.php, 单 击 【 转 到 】 按 钮 。 Eees yeep pap 
会 显示 如 图 3-5 所 示 的 窗口 。 了 件 四 久久 区) 查看 0 收 席 0 工具 全) 宰 助 Qi) 


二 站 四 | 天 ae-ymlyeaberctos0/mmvetyaa php 届 贺 和 到 让 > 


在 本 案例 中 ， 创 建 了 一 个 函数 cook0， 该 函数 返回 “| 
指定 字符 串 的 值 。 创 建 函 数 时 在 小 括号 中 创建 了 一 个 字 | xf 
符 串 变量 。 在 下 面 调用 时 ,如果 没 有 给 该 函数 传递 参数 ， 
直接 使 用 默认 值 就 可 以 了 ， 如 果 使 用 了 参数 ， 函 数 就 会 - 
使 用 指定 的 参数 值 。 在 使 用 默认 参数 时 需要 注意 ， 默 认 国生 全 泊 

值 必须 是 常量 表达 式 ， 不 能 是 变量 。 如 果 函 数 有 多 个 参数 ， 可 以 为 多 个 参数 指定 默认 值 。 但 需要 注 
意 ， 为 了 避免 二 义 性 ， 在 指定 了 一 个 默认 值 的 参数 的 右边 ， 不 能 出 现 没有 指定 默认 值 的 参数 。 如 下 
面 的 示例 就 是 错误 的 : 


function test ($x, $y=1, $2){ 


} 
test (8,3); 


在 上 述 代码 中 ， 如 果 使 用 test(8,3) 调 用 函数 test0， 将 会 出 现 二 义 性 错误 ， 因 为 无 法 确定 3 应 该 
传递 给 y 还 是 z， 可 以 将 上 述 代 码 修正 为 : 


function test ($x,$y,$z=1) { 


h 
test(873)5 


3.2.5 ”可 选 参数 


在 PHP 4.0 中 还 新 增 了 一 种 参数 传递 的 方式 一 一 可 变 参数 列表 。 可 以 在 自 定义 的 函数 中 依次 将 
需要 传递 的 参数 一 一 列 出 ， 然 后 使 用 指定 的 函数 来 获得 参数 ， 这 些 函 数 如 表 3-1 所 示 。 


表 3-1 可 选 参 数 函 数 


名 称 说 了 明 
func_num argsO 返回 自 定义 的 函数 中 传 入 的 参数 个 数 。 
fune setarg0 取得 第 arg_num+1 个 参数 的 值 : 
fanc_get args0 返回 一 个 包含 所 有 参数 的 数组 “oo? 
下 面 创建 一 个 案例 ， 演 示 可 选 参 数 的 使 用 。 该 案例 的 代码 如 下 所 示 : 
案例 3-6 
<?php 


function test(){ 
$num=func num args(); 
echo "参数 的 个 数 为 :$num<br>"; 
if ($num>=3){ 
$test=func get arg(2); 
echo "第 三 个 参数 是 : S$test<br>"; 
| 
Stest=func get args ()7 
for ($i=075Si<SnumzSi++)T{ 
echo "第 $i"; 
echo "个 参数 是 : $test [$i] <br>"; 
} 
test(lLll 22733.44)? 
2 


将 上 述 代码 保存 ， 文 件 名 为 arg4.php， 并 保存 到 指定 的 位 置 。 打 开 正 浏览 器 ， 在 地 址 栏 中 输入 
http://localhost:8080/myweb/arg4.php, 单 击 【 转 到 】 按 钮 ， 
ER 会 显示 如 图 3-6 所 示 的 窗口 。 

在 本 案例 中 ,创建 了 一 个 函数 test0， 该 函数 没有 参 
数 ， 使 用 可 选 参数 。 利 用 func_num args0 函 数 获得 可 选 
参数 的 个 数 ， 并 判断 函数 的 可 选 参 数 的 个 数 ， 使 用 
func_get_argO 函 数 获得 指定 位 置 的 数值 ， 然 后 输出 。 最 
后 使 用 func _get_argsO 函 数 获得 所 有 的 可 选 参数 , 并 以 数 
组 的 形式 返回 。 用 for 循环 遍历 输出 每 个 可 选 参数 。 3-6 可 选 参数 


3.2.6 ”从 函数 返回 值 


在 PHP 技术 中 ， 函 数 是 非常 重要 的 ， 可 以 执行 一 个 功能 操作 ， 或 者 返回 一 个 数值 。 当 函数 具有 
返回 值 后 ， 就 可 以 传递 信息 给 函数 的 调用 者 。 在 PHP 的 函数 中 ， 可 以 使 用 retum 关键 字 返 回 一 个 
数值 。 

returnO 语 句 可 以 向 函数 调用 者 返回 任意 确定 的 值 ， 将 程序 控制 权 返 回 到 调用 者 的 作用 域 。 如 果 
returnO 语 名 在 全 局 作用 域内 调用 ， 将 中 止 脚本 的 执行 。 其 使 用 示例 如 下 所 示 : 
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返回 


比如 
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<?php 
function div($a, $b){ 
return $a/$b; 
} 
function str($string){ 
$string.=Date('Y-M-D'); 
return S$string; 
} 
$d=div (6,8); 
echo "$d"; 
echo("<hr>"); 
$e=str ("现在 的 日 期 ") ; 
Cecho He 
2 


在 上 述 代 码 中 ， 共 创建 了 两 个 函数 ， 分 别 为 div0 和 str0。 在 div0 函 数 中 直接 使 用 return0 语 句 
两 个 参数 相 除 的 结果 。 在 第 二 个 函数 中 ， 返 回 的 是 已 经 操作 过 的 字符 串 。 

从 函数 中 也 可 以 一 次 性 返回 多 个 值 。 例 如 ， 假 设 要 创建 一 个 函数 ， 从 数据 库 中 获取 用 户 数据 ， 
] 户 的 姓名 、 电 子 邮 件 地 址 和 电话 号 码 ， 然 后 返回 给 调用 者 。 完 成 这 个 任务 比较 简单 ， 使 用 一 


有 用 的 语言 构造 list0 就 可 以 实现 。 利 用 list0 构 造 可 以 很 方便 地 从 数组 中 获取 值 。 
下 面 创 建 一 个 案例 ， 演 示 使 用 函数 返回 多 个 值 。 该 案例 的 代码 如 下 所 示 : 

案例 3-7 

<?php 


function test(){ 
$use[]=" 刘 华 林 "; 
$use[]="liuhaihua@sohu.com"; 
Suse[]=" 男 "7 
return S$use; 
1 
1ist(Sname, $email, $sex)=test (); 
echo "姓名 :Sname, 电子 邮件 :$email, 性别:$sex"; 


?> 


将 上 述 代 码 保存 ， 文 件 名 为 rrtphp， 并 保存 到 指定 


的 位 置 ,打开 正 浏 览 器 ,在 地 址 栏 中 输入 http://localhost: | wetovreere ne 
8080/myweb/ret.php, 单 击 【 转 到 】 按 钮 , 会 显示 如 图 3-7 | 诗 名 : 询 华 宰 , 电 子 闻 件 :liwhaihuaschu. com 性 别 : 另 
所 示 的 窗口 。 

EEC 
3.2.7 ”其 套 函 数 图 3-7 返回 多 个 值 


在 PHP 中 还 支持 谋 套 函数 ， 顾 名 思 义 就 是 在 一 个 函数 中 定义 ， 并 调用 另外 一 个 函数 。 如 果 一 个 


函数 的 代码 或 者 操作 过 多 ， 可 以 再 创建 一 个 函数 完成 部 分 功能 。 


下 面 创建 一 个 案例 ， 演 示 一 下 嵌 套 函数 的 使 用 。 该 案例 的 代码 如 下 所 示 : 


案例 3-8 
<?php 
function sss($a){ 
function st($a){ 
$sum=$a+7; 
return $sum; 
} 
if($a>5) 
{echo "$a";} 
elsef{ 
$sum=$at+st (4); 
echo "结果 为 $sum"; 
} 
} 
sss(3); 
?> 


将 上 述 代码 保存 ， 文 件 名 为 qtphp， 并 保存 到 指定 的 位 置 。 打 开 正 浏览 器 ， 在 地 址 栏 中 输入 
http://localhost:8080/myweb/qt.php， 单 击 【 转 到 】 按 钮 ， 会 显 
示 如 图 3-8 所 示 的 窗口 。 

在 本 案例 中 ， 创 建 了 一 个 函数 sss0， 用 来 求 取 一 个 数值 。 
该 函数 具有 一 个 参数 a， 如 果 该 参数 的 值 大 于 5， 就 直接 输出 
该 值 。 如 果 参 数 的 值 小 于 5， 就 会 调用 另外 一 个 函数 st0，stO 
函数 是 在 函数 sss0 中 定义 的 一 个 函数 ， 该 函数 只 有 一 个 参数 ， 
其 功能 是 把 参数 的 值 和 指定 的 数 相 加 , 并 使 用 retum 语句 返回 
该 值 。 当 在 函数 sssO 中 调用 stO 函 数 时 , 把 stO 函 数 的 返回 值 和 参数 a 的 值 相 加 , 然后 调用 sss0 函 数 ， 
并 输出 结果 值 。 


3.2.8 ”递归 函数 


图 3-8 媒 套 函数 


递归 函数 (recursive function) 即 调用 自身 的 函数 , 这 对 于 程序 员 来 说 , 通常 有 很 高 的 实用 价值 ， 
常用 来 将 复杂 的 问题 分 解 为 简单 的 并 相同 的 情况 ， 反 复 做 这 种 处 理 直 到 问题 解决 ， 如 在 菜单 中 创建 
菜单 项 等 。 

下 面 创建 一 个 案例 ， 演 示 递 归 函 数 的 创建 和 调用 。 该 案例 的 代码 如 下 所 示 : 


案例 3-9 
<hl align=center> 递 归 函 数 使 用 示例 </h1> 
<?php 
function ab($a){ 
if($a>10) 
{echo ($a);} 
else 
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{ echo ($a); 

echo ("gnbsp;&nbsp;"); 
$a=$a+tl; 

ab($a); 

和 

} 
ab (3) 7 
?> 


将 上 述 代 码 保存 ， 文 件 名 为 digui.jphp， 并 保存 到 指定 的 位 置 。 打 开 正 浏览 器 ， 在 地 址 栏 中 输 
入 http://localhost:8080/myweb/digui.jphp， 单 击 【 转 到 】 
按钮 ， 会 显示 如 图 3-9 所 示 的 窗口 。 

在 本 案例 中 , 创建 了 一 个 函数 ab0， 在 该 函数 体 中 ， 
对 函数 参数 进行 判断 ， 如 果 函 数 参 数 的 值 大 于 10， 就 输 
出 该 值 ， 如 果 参 数 的 值 小 于 10， 就 输出 该 值 ， 并 使 参数 
的 值 在 原来 的 基础 上 加 1， 以 变化 后 的 参数 为 参数 重新 
调用 ab0 函 数 。 在 下 面 的 例子 中 ， 调 用 该 函数 并 传递 一 图 3-9 递归 函数 使 用 
个 参数 3。 

使 用 递归 策略 通常 能 大 幅 减 少 代 码 量 ， 提 高 重用 性 。 虽 然 递归 函数 并 不 总 是 最 优 的 解决 方案 ， 
但 它 对 于 任何 语言 来 说 都 是 有 益 的 补充 。 


3.2.9 变量 函数 


递归 函数 使 用 示例 


7 8 9 10 11 


在 编写 程序 代码 时 ， 如 果 要 处 理 一 个 比较 复杂 的 问题 ， 就 会 在 程序 中 创建 多 个 函数 。 为 了 调用 
每 个 函数 ， 需 要 记 住 每 个 函数 的 名 称 和 相应 的 功能 。 一 般 情况 下 ， 要 做 到 函数 的 名 称 见 文 知 义 ， 即 
见 到 函数 名 就 知道 完成 什么 样 的 功能 。 然 后 在 不 同 的 环境 下 调用 这 些 函 数 。 在 PHP 中 ,还 可 以 通过 
一 种 简短 得 多 的 方式 完成 同样 的 目标 : 变量 函数 ， 变 量 函 数 variable function) 是 指 这 个 函数 名 也 
要 在 执行 前 计算 ， 这 意味 着 函数 名 直到 执行 时 才 确 定 ， 就 像 正常 的 变量 一 样 ， 变 量 函 数 前 面 有 个 美 


元 符号 。 
下 面 创建 一 个 案例 ， 演 示 一 下 变量 函数 的 创建 和 使 用 。 该 案例 的 代码 如 下 所 示 : 
案例 3-10 
<?php 


function funcl(){ 
echo "www.itzcn.com 是 一 个 IT 学 习 网 站 <br>"; 
} 
function func2(){ 
echo "要 学 习 电脑 知识 ， 请 登录 该 网 站 <br>"; 
} 
$templ="funcl"; 
Stemp2="func2"7 
Stempl1 () 7 
Stemp2 ()7 
> 


将 上 述 代码 保存 ， 文 件 名 为 bian php， 并 保存 到 指定 的 位 置 。 打 开 正 浏览 器 ， 在 地 址 栏 中 输入 
http://localhost:8080/myweb/bian.php， 单 击 【 转 到 】 按 钮 ,会 


ET 


显示 如 图 3-10 所 示 的 窗口 。 人 ET 


在 本 案例 中 ， 创 建 了 两 个 函数 func10 和 func20， 这 两 
个 函数 分 别 输出 一 个 字符 串 信 息 。 使 用 语句 
“$temp1="func1";” 把 创建 的 函数 和 变量 连接 起 来 ， 在 下 面 
调用 该 函数 时 就 可 以 直接 以 变量 的 名 称 调用 该 函数 。 
虽然 有 时 候 使 用 变量 函数 很 方便 ， 但 是 要 注意 ， 它 们 存 图 3-10 变量 函数 

在 安全 风险 。 最 显著 的 问题 是 ， 攻 击 者 可 能 会 修改 用 来 声明 函数 名 的 变量 ， 以 此 来 执行 PHP 的 任何 
函数 。 例 如 ， 如 果 将 前 面 示例 中 的 templ 变量 设置 为 exec， 将 temp2 变量 设置 为 mm - rf， 那 么 PHP 
的 exec0 命 令 将 “放心 地 ”在 系统 级 执行 其 参数 。 而 命令 rm - 王将 从 根 目 录 开始 ， 递 归 地 删除 所 有 
文件 。 最 终 的 结果 将 是 灾难 性 的 。 因 此 ， 一 定 要 确保 过 滤 所 有 用 户 信息 ， 否 则 ， 永 远 不 知道 接 下 来 
会 发 生 什么 。 


. 
. 
. 
3 习 J a pA 
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在 可 变 函 数 名 的 使 用 中 需要 注意 ， 可 变 函 数 名 与 普通 函数 调用 时 的 最 大 区 别 在 于 “$"”， 普 通 函数 在 调 
提示 用 时 不 需要 加 “$”， 而 调用 可 变 函 数 名 的 函数 ， 在 该 变量 之 前 必须 要 有 “$"。 


3.3 ”函数 库 


在 PHP 技术 中 ， 提 供 多 种 类 型 的 内 置 函 数 ， 如 文件 函数 、 数 据 库 函 数 、 数 学 函数 等 。 这 样 可 以 
节省 处 理 问题 时 编写 重复 的 代码 。 函 数 的 使 用 ， 达 到 了 代码 重用 的 目的 。 同 样 也 可 以 自 定义 函数 ， 
放 入 函数 库 中 作为 自己 的 内 置 函 数 使 用 。 本 节 将 首先 介绍 一 下 常用 的 数学 函数 和 时 间 函 数 ， 然 后 介 
绍 一 下 如 何 创建 自己 的 函数 库 。 


3.3.1 Math 数学 函数 


在 标准 函数 库 中 ， 数 学 函数 是 其 中 的 一 个 类 别 ， 这 些 数学 函数 仅 能 处 理 在 计算 机 上 integer 和 
float 范围 内 的 值 (目前 这 对 应 于 C 语言 中 的 long 和 double)。 使 用 该 类 别 函数 无 须 安装 、 编 译 、 配 
置 ， 直 接 使 用 就 可 以 了 ， 因 为 数学 函数 是 PHP 内 核 的 一 部 分 。 

在 这 里 可 以 把 Math 作为 一 个 对 象 来 看 待 ， 因 为 它 不 但 具有 一 些 常量 ， 还 具有 方法 。 其 常量 
M_PI 表示 圆周 率 ，M_E 表示 指数 等 。Math 常用 的 方法 如 表 3-2 所 示 。 

表 3-2 Math 常用 方法 


找 出 最 小 值 


asin0) | 反正 弦 指数 表达 式 

ceil0 | 进 一 法 取 整 产生 一 个 随机 整数 
decbinO | 十 进 制 转换 为 二 进 制 对 浮 点 数 进行 四 舍 五 入 
floorO | 舍 去 法 取 整 正弦 


找 出 最 大 值 平方 根 
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下 面 创建 一 个 案例 ， 演 示 一 下 数学 函数 的 使 用 方法 。 该 案例 的 代码 如 下 所 示 : 
案例 3-11 
<?php 
echo "使 用 rand 函数 产生 一 个 随机 数 "; 
echo rand(); 
echo "<br> 使 用 round 函数 求 取 一 个 整数 :"; 
print round(3.55); 
echo "<br> 使 用 floor 函数 求 取 一 个 整数 "; 
echo floor(3.55); 
echo "<br> 使 用 ceil 函数 求 取 一 个 整数 "; 
Echo ceil(3.55)? 
echo "<br> 使 用 max 函数 求 取 两 个 数 的 最 大 值 "; 
echo max(3,8); 
echo "<br> 使 用 sqrt 函数 求 取 一 个 数 的 平方 根 "; 
echo sqrt(4); 
?> 


将 上 述 代 码 保存 ， 文 件 名 为 Math php， 并 保存 到 指 “ET 
定 的 位 置 。 打 开 正 浏览 器 ， 在 地 址 栏 中 输入 | 汪汪 证 区 由 下 时 
http://localhost:8080/myweb/Math.php， 单 击 【 转 到 】 按 | 各 大 r=sa 画 数 产 生 -rs 
钮 ， 会 显示 如 图 3-11 所 示 的 窗口 。 yi 人 

在 本 案例 中 ， 调 用 了 多 个 数学 函数 进行 运算 。 需 要 | 串 训 总 二 级。 
注意 的 是 ，round0 函 数 、floor0 函 数 和 ceilO 函 数 都 是 求 
取 一 个 浮 点 数 的 整数 ， 但 是 在 求 整 的 过 程 中 ， 采 用 的 规 
则 不 一 样 。 图 3-11 Math 常用 方法 


3.3.2 ”日 期 /时 间 函 数 


在 PHP 中 , 存在 一 类 日 期 /时 间 函 数 , 可 以 用 这 些 函数 得 到 PHP 所 运行 的 服务 器 的 日 期 和 时 间 ， 
并 可 以 用 这 些 函数 将 日 期 和 时 间 以 很 多 不 同方 式 格式 化 输出 。 本 节 将 一 一 介绍 常用 的 日 期 /时 间 函 
数 。 这 些 函 数 的 详细 信息 如 表 3-3 所 示 。 


表 3-3 日 期 /时 间 函 数 


名 称 示 例 

checkdate($month. 如 果 应 用 的 值 构成 一 个 有 效 日 期 , 则 | <?php 

$date.$year) 该 函数 返回 为 真 。 例如 ,对 于 错误 日 | /返回 值 为 假 
期 2005 年 2 月 31 日 ,此 函数 返回 为 | echo checkdate(2.31.2008) ? "valid" : "invalid": 
假 。 在 日 期 用 于 计算 或 保存 在 数据 库 | /返回 值 为 真 
中 之 前 , 可 用 此 函数 检查 日 期 并 使 日 | echo checkdate(4.6.2010) ? "valid" : "invalid" 
期 生效 

getdate($ts) 在 没有 自 变量 的 情况 下 , 该 函数 以 结 | <?php 


合 数组 的 方式 返回 当前 日 期 与 时 间 。| /得 到 的 日 期 是 一 个 数组 


名 称 EE 
数组 中 的 每 个 元 素 代表 日 期 /时 间 值 | $arr = getdate0: 
中 的 一 个 特定 组 成 部 分 。 可 向 函数 提 | echo "日 期 是 " . Sarr[mday'] . " " . $am['weekday'] . 本 
交 可 选 的 时 间 标签 自 变量 , 以 获得 与 | " " . Sarr[year]: 。 ; . 


时 间 标 签 对 应 的 日 期 /时 间 值 。 应 用 
此 函数 来 获得 一 系列 离散 的 、 容 易 分 
离 的 日 期 /时 间 值 


echo "时 间 是 " . Sarr[hours'] . ":" . Sarr[minutes']; 
?> 


date($format $ts) 可 用 在 一 系列 的 修正 值 中 , 将 整数 时 
间 标 签 转变 为 所 需 的 字符 串 格式 


<2php 

/ 格式 化 当前 日 期 

// 返 回 值 为 "13-Sep-2008 01:16 PM" 

echo date("d-M-Y h:iA" mktimeO): 

?> 

<?php 

// 返 回 值 13-Sep-05 

echo date("d-M-y", strtotime("today")): 

// 返 回 值 14-Sep-05 

echo date("d-M-y". strtotime("tomorrow")): 
// 返 回 值 16-Sep-05 

echo date("d-M-y", strtotime("today +3 days")); 
?> 

<?php 

// 获 得 开始 时 间 

$start = microtime(): 

// 执 行 代码 

for ($x=0: $x<1000: $x++) { 

Snull = $x * $x: 

} 

// 获 得 结束 时 间 

$end = microtimeO: 

// 计 算 二 者 的 时 间 差 

echo "Elapsed time: " . ($end - $start) ." sec": 
?> 

<?php 

/转换 当前 的 时 间 到 GMT 格式 
/返回 值 为 "13-Sep-2005 08:32 AM" 

echo gmdate("d-M-Y h:i A". mktimeO): 

?> 


strtotime($str) 此 函数 将 可 阅读 的 英文 日 期 /时 间 字 
符 串 转换 成 UNIX 时 间 标 签 。 应 用 此 
函数 将 非 标准 化 的 日 期 /时 间 字 符 串 
转换 成 标准 、 兼 容 的 UNIX 时 间 标 签 
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microtime() 将 UNIX 时 间 标 签 格式 化 成 适用 于 
当前 环境 的 日 期 字符 串 


gmdate($formmat $ts) 此 函数 将 UNIX 时 间 标 签 格式 化 成 
可 人 为 阅读 的 日 期 字符 串 。 此 日 期 字 
符 串 以 GMT ( 非 当 地 时 间 ) 表示 


date_default timezone ”| 此 函数 此 后 所 有 的 日 期 /时 间 函 数 调 | <?php 

set($tz) 用 设 定 并 恢复 默认 的 时 区 /设置 时 区 

date_default timezone date_default timezone set(UTC"): 
_getO ?> 
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date_default timezone set($tz) 函 数 仅 在 PHP 5.1+ 中 有 效 。 此 函数 是 一 个 方便 的 捷径 ， 可 为 以 后 的 时 间 
提示 操作 设 定时 区 。 


3.3.3 ” 自 定义 函数 库 


创建 一 个 Web 项 目 ， 可 能 需要 在 一 个 Web 页 面 创建 多 个 自 定义 函数 来 完成 指定 的 功能 。 如 果 
在 别 的 Web 页 面 也 需要 完成 相同 的 功能 ， 就 需要 创建 这 些 相同 的 函数 ， 再 调用 这 些 函 数 。 假 如 多 个 
Web 页 面 都 需要 调用 这 些 函 数 ， 如 果 在 每 个 页 面 都 创建 这 些 函 数 ， 这 需要 做 大 量 的 重复 工作 ， 这 时 
通常 将 函数 组 织 到 函数 库 〈library) 中 〈 即 一 个 PHP 页 面 中 )， 这 样 就 方便 了 其 他 的 PHP 页 面 调用 
这 些 函 数 ， 达 到 代码 重用 的 目的 。 在 一 个 文件 中 简单 地 聚集 函数 定义 就 可 以 创建 PHP 库 。 其 语法 格 
式 如 下 所 示 : 


保存 这 个 库 , 最 好 使 用 一 个 能 清楚 地 说 明 其 用 途 的 命名 约定 来 命名 ， 如 taxes.library.php。 然 后 ， 
使 用 include0、include_ once0、require0 或 require_onceO 将 这 个 函数 库 插入 脚本 中 ，( 另 外 ， 也 可 以 
使 用 PHP 的 auto_prepend 配置 指令 自动 完成 文件 插入 ) 例如 , 假设 将 这 个 库 命名 为 taxes.library.php， 
可 以 如 下 将 其 包含 到 脚本 中 : 


下 面 创建 一 个 案例 ， 演 示 一 下 自 定义 函数 库 的 创建 和 使 用 。 首 先 创建 要 使 用 的 函数 ， 其 代码 形 
式 如 下 所 示 : 


将 该 文件 保存 ,文件 名 为 Function php， 并 保存 到 C:\Web\vapache\htdocswmyweb 目录 下 。 下 面 创 
建 调 用 这 些 PHP 函数 的 页 面 ， 其 代码 如 下 所 示 : 


案例 3-12 

<?php 
require once("Function.php"); 
echo "8 和 2 相 除 的 结果 为 : "; 
echo ful(8,2); 
echo "<br> 现 在 的 日 期 为 : "; 
fu2(); 

A 


将 该 文件 保存 ， 文 件 名 为 Fu.php， 并 保存 到 C:\Web\apache\htdocsmyweb 目录 下 。 打 开 正 浏览 
器 ， 在 地 址 栏 中 输入 http://localhost:8080/myweb/Fu.php， 单 击 【 转 到 】 按 钮 ， 会 显示 如 图 3-12 所 示 
的 窗口 。 


到 http localhost: 8080/myveb/Fu. php — Wicrosoft In. 
文件 5) 编 洋 里 ] 查看 0) 收容 W] 工具 IJ 帮助 0D 
Ht Ee 2 oo/ re ls] 


8 和 和 2 相 除 的 结果 为 ，4 
羽 在 的 日 期 为 ，2007-Jun-Thu 


图 3-12 自 定 义 函 数 库 


从 本 案例 中 可 以 看 出 ， 当 使 用 语句 “require_once("Function.php");” 包 含 一 个 页 面 到 当前 页 面 ， 
当前 页 面 Fu 就 可 以 直接 使 用 Function 页 面 中 创建 的 函数 了 。 
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学 习 目标 |oujecuve 


在 PHP 的 程序 设计 中 ， 如 果 要 同时 记录 多 项 数据 列 〈 或 称 为 数据 集 )， 使 用 变量 是 一 个 可 行 的 
方法 ， 但 在 处 理 需 要 进行 分 类 的 信息 时 ， 如 学 生 基 本 信息 、 各 科 成 绩 信息 或 在 校 表现 信息 时 ， 需 要 
在 程序 中 使 用 很 多 变量 , 这 种 情况 将 会 导致 程序 设计 混乱 ， 所 以 PHP 与 ASP、Java 等 语言 一 样 也 引 
进 了 数组 (array)。 通 过 数组 程序 人 员 可 以 很 方便 地 解决 上 述 问题 。 


LETPR 
2 内 容 摘要 |avstrsct 


理解 什么 是 数组 
掌握 如 何 创建 数组 

掌握 如 何 输 出 及 测试 数组 
熟练 掌握 增加 和 删除 数组 元 素 
熟练 掌握 定位 数组 元 素 
掌握 确定 数组 大 小 和 唯一 性 
熟练 掌握 数组 的 应 用 

了 解 其 他 数组 函数 


4.1 初 识 数组 


数组 提供 了 一 种 理想 的 方法 来 存储 、 操 作 、 排 序 和 获取 数据 集 。PHP 的 解决 方案 也 不 例外 ， 它 
也 支持 数组 数据 类 型 ， 还 支持 与 数组 操作 有 关 的 大 量 函 数 。PHP 中 提供 了 30 多 个 数组 相关 函数 ， 
其 中 很 多 通用 函数 允许 用 户 检查 给 定数 组 中 是 否 存在 特定 对 象 、 对 数组 元 素 计数 、 增 加 或 删除 元 素 ， 
以 及 对 元 素 排序 等 。 


4.1.1 什么 是 数组 


通常 情况 下 ， 数 组 (array) 被 定义 为 一 组 具有 某 种 共同 特性 的 元 素 ， 如 相似 性 〈 车 、 球 、 水 果 
类 型 等 ) 和 类 型 〈 例 如 所 有 元 素 都 是 字符 串 或 整数 ) 的 集合 ， 集 合 中 的 所 有 元 素 之 间 由 一 个 特殊 的 
标识 符 来 区 分 ， 该 标识 符 通常 被 称 为 键 (key)。 而 在 PHP 语言 中 这 种 定义 有 所 改变 ， 因 为 PHP 数 
组 结构 中 可 以 包括 完全 无 关 的 元 素 ， 也 可 以 包含 不 属于 同一 种 类 型 的 元 素 。 例 如 ， 一 个 数组 可 能 包 
括 学 生 的 姓名 、 性 别 、 联 系 电话 、 考 试 成 绩 或 驾驶 证 编号 等 一 些 不 相关 的 元 素 。 

1 何 时 使 用 数组 

通常 利用 单个 变量 或 数组 就 可 以 记录 某 个 或 某 些 数据 ， 那 么 如 何 判 定 何 时 使 用 数组 呢 ? 假如 要 


记录 一 个 学 生 〈Jack) 的 基本 信息 及 各 项 成 绩 ， 所 需 的 字段 有 学 号 、 姓 名 、 性 别 、 系 别 、 班 级 、 身 
份 证 号 、 汇 编 成 绩 、 英 语 成 绩 及 数学 成 绩 等 。 如 果 用 变量 来 进行 记录 ， 就 需要 定义 9 个 变量 ， 如 果 
要 记录 的 字段 不 只 有 9 个 ， 有 几 十 、 甚 至 上 百 个 ， 那 么 结果 是 可 想 而 知 的 。 此 时 若 使 用 数组 就 比较 
简单 ， 只 需要 定义 一 个 数组 变量 $jack 就 可 以 了 。 

另外 ， 如 果 要 记录 光标 在 屏幕 上 的 位 置 ， 同 样 可 以 使 用 一 个 数组 变量 $xy 来 存储 水 平 坐标 x 及 垂 
直 坐标 y 的 值 ， 而 不 使 用 两 个 变量 gx、$y。 所 以 在 记录 多 项 信息 时 ， 使 用 数组 的 优势 是 相当 明显 的 。 

2. PHP 数组 的 优点 

PHP 中 提供 的 数组 与 其 他 语言 (如 Java、C 语言 ) 中 提供 的 数组 最 大 的 不 同 是 ,在 PHP 中 同一 
数组 中 的 元 素 可 以 是 不 同类 型 的 值 ,这 些 元 素 之 间 可 能 无 任何 联系 假设 这 里 定义 了 一 个 数组 Yan[]， 
它 的 值 如 下 所 示 : 

$Yan[0]=" 宋 岩 岩 "; 

$Yan[1]="13705211314"; 

$Yan[2]=99; 

$Yan[3]=96; 

$Yan[4]=" 北 京 "; 

$Yan[5]="IT 在 中 国 : http://www.itzcn.com"; 

上 述 数 组 Yan[] 中 索引 号 为 0、1、4、5 的 元 素 所 存储 的 姓名 、 联 系 方式 、 籍 贯 及 网 址 ， 其 类 型 
都 为 字符 串 ; 而 索引 号 为 2、3 的 元 素 所 存储 的 是 数学 成 绩 与 计算 机 成 绩 ， 其 类 型 都 是 整数 ; 在 这 个 
数组 中 索引 号 为 5 的 元 素 值 与 前 面 其 他 元 素 值 没有 任何 关系 , 但 它 同样 存储 在 该 数组 中 。 在 PHP 中 ， 
通常 使 用 数组 元 素 存储 不 同类 型 的 数据 ， 这 为 处 理 数据 及 定义 数据 类 型 提供 了 方便 。 


4.1.2 ”创建 数组 


在 PHP 中 ， 数 组 在 使 用 之 前 并 不 需要 声明 ， 它 的 声明 将 在 初始 化 时 自动 完成 。 在 PHP 中 创建 
数组 主要 有 两 种 方法 ， 如 表 4-1 所 示 。 


表 4-1 创建 数组 的 方法 


方 法 说 阴 
赋值 以 索引 方式 给 数组 变量 赋值 
arrayO 通过 调用 array0 函 数 来 实现 


下 面 分 别 对 表 4-1 所 示 的 两 种 方法 进行 详细 介绍 。 

1. 使 用 赋值 创建 数组 

在 PHP 中 ， 创 建 数组 最 简单 的 方法 就 是 使 用 赋值 ， 该 方法 的 语法 格式 如 下 所 示 : 
Svariable [<key expr>] = <expr>; 

根据 上 述 语法 格式 来 看 如 下 所 示 的 简短 例子 是 如 何 实现 的 : 


$StuName[0]=" 宋 岩 岩 "; 
SStuName [1]=" 龙 建华 "; 


变量 名 后 面 的 方 括号 “[]” 向 PHP 指出 变量 StuName[] 是 一 个 数组 变量 。 PHP 将 在 从 StuName[0] 
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开始 的 连续 编号 的 单元 中 存储 数组 的 值 。 除 了 上 述 形式 ， 用 户 还 可 以 编写 一 个 指定 标识 符 值 的 赋值 
语句 ， 这 样 就 可 以 将 一 个 值 与 一 个 特定 的 数组 元 素 相 关 ， 这 种 形式 的 示例 如 下 所 示 : 

$StuName[0]=" 宋 岩 岩 "; 

SStuName [1]=" 龙 建华 "; 

$stuName[]=" 王 晶 晶 "; 

这 里 需要 注意 的 是 , 最 后 的 赋值 不 包括 索引 号 。 PHP 自动 将 值 与 数组 的 下 一 个 连续 元 素 相 关联 ， 
所 以 上 述 值 “ 王 晶 晶 ” 在 这 个 数组 中 的 索引 编号 为 2。 

上 述 数 组 的 索引 号 是 连续 的 , 其 实 PHP 中 数组 元 素 并 不 需要 与 连续 的 标识 符 (索引 号 ) 相关 联 。 
这 种 形式 的 创建 如 下 所 示 : 

SStuName [10]=" 宋 岩 岩 "; 


SStuName [100]=" 龙 建华 "; 
$stuName[]=" 王 晶 晶 "; 


由 于 PHP 并 不 限制 一 定 要 使 用 整数 作为 标识 符 , 所 以 通过 将 字符 串 指 定 为 标识 符 也 可 以 创建 一 
个 关联 数组 ， 也 称 为 字符 串 索引 数组 。 这 种 形式 的 创建 如 下 所 示 : 

$sStuName["BoB"]=" 宋 岩 岩 "; 

SStuName ["Jack"]=" 龙 建华 "; 

$stuName["Fri"]=" 王 晶 蝇 "; 

通常 情况 下 还 是 更 多 地 使 用 整数 作为 标识 符 ， 因 为 这 样 ， 在 实际 应 用 中 用 户 可 以 使 用 for 循环 
在 数组 内 进行 迭代 ， 也 就 是 检查 每 一 个 元 素 值 。 

2. 使 用 array() 函 数 创建 数组 

除了 使 用 赋值 语句 外 ,还 可 以 使 用 arrayO 函 数 创建 PHP 数组 。 这 里 可 以 将 amrayO 函 数 想象 为 包 
含 一 长 串 元 素 的 数组 ， 各 元 素 以 “,” 分 隔 。 该 方法 的 语法 格式 如 下 所 示 : 


卫 
Sa 
卫 
十 

三 
Dake 

ol 
名 
| 
完 
宇 
过 
习 
手 
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$variable = array([mixed ...]); 

根据 上 述 语法 格式 来 看 如 下 所 示 的 简短 例子 是 如 何 实现 的 : 

$StuName=array (" 宋 岩 岩 "，" 龙 建华 ") ; 

如 果 要 将 一 个 特定 的 标识 符 与 一 个 值 相关 联 , 则 可 以 使 用 => 运 算 符 。 这 种 形式 的 示例 如 下 所 示 : 
$stuName=array (" 宋 岩 岩 "，" 龙 建华 ", 100=>" 王 晶 晶 ") ; 


注意 ， 与 一 个 值 相关 的 标识 符 并 不 一 定 是 连续 的 ， 也 不 一 定 是 整数 ， 它 也 可 以 为 字符 串 。 这 种 
形式 如 下 所 示 : 
$stuName=array ("BoB"=>" 宋 岩 岩 "，" 龙 建华 ",100=>" 王 晶 晶 ") 7 


4.1.3 输出 及 测试 数组 


创建 完 数组 后 就 可 以 对 其 进行 一 些 操作 ， 比 如 输出 数组 中 各 元 素 的 值 、 测 试 某 一 变量 是 否 为 数 
组 等 ， 如 表 4-2 所 示 。 


表 4-2 输出 及 测试 数组 的 函数 


说 明 
该 函数 将 按照 一 定格 式 显示 数组 的 索引 和 元 素 


使 用 内 置 函 数 is_array0 来 判定 某 个 特定 变量 是 否 为 一 个 数组 


下 面 详细 介绍 如 何 输出 数组 以 及 测试 数组 。 
1. 输出 数组 
输出 数组 主要 用 到 print_r0 函 数 ， 代 码 如 下 所 示 : 


案例 4-1 

<pre> 

<?php 
$stuName=array ("BoB"=>" 打 岩 岩 ", "Jack"=>" 龙 建华 ", "Friends"=>array(" 王 晶 晶 "," 赵 星 
"," 李 晓 渊 ") ) ; 
print Fr (SStuName) 7 

?> 2 

</pre> 


将 上 述 代 码 存储 在 4-1.php 文件 中 , 保存 到 Apache2.2\htdocs 目录 下 的 子 目录 4 中 , 然后 打开 下 
浏览 器 ， 在 地 址 栏 中 输入 http:Wlocalhost/4/4-1.php， 运 行 结 果 如 下 所 示 : 
案例 4-1 


Array 
( 
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[BoB] => 宋 岩 岩 
[Jack] => 龙 建华 
[Friends] => Array 
( 
[0] => 王 晶 晶 
[1] => 赵 星 
[2] => 李 晓 渊 


) 


当 print_r0 函 数 将 数组 的 指针 移 到 最 后 时 , 可 以 使 用 resetO 函 数 让 指针 回 到 开始 处 。 如 果 想 返回 
print_r0 函 数 的 输出 ， 可 使 用 retum 参数 。 若 此 参数 设 为 Tme，print_r0 函 数 将 不 打印 结果 (此 为 默 
认 操 作 )， 而 是 返回 其 输出 。 

2. 测试 数组 

测试 数组 函数 is_array0 的 具体 格式 如 下 所 示 : 


boolean is array (mixed variable) 


is_array0O 函 数 用 来 判定 variable 是 否 为 数组 ， 如 果 是 则 返回 True， 否 则 返回 False。 这 里 需要 注 
意 的 是 ， 即 使 数组 只 包含 一 个 值 ， 也 将 被 认为 是 一 个 数组 。 它 的 使 用 方法 如 下 所 示 : 


案例 4-2 
<?php 
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SStuNamel = array ("BoB” => “" 宋 岩 岩 "，"”Jackn => " 龙 建 华 '", "Fri” =>“ 王 晶 晶 ") > 
$sStuName2 一 " 宁 岩 岩 "; 
echo "\$stuNamel is an array:".is array(SStuName1l) ."<br/>"7 
echo "\$stuName2 is an array:" -is array($StuName2)."<br/>"; 
nS 


将 上 述 代 码 存 储 在 4-2.php 文件 中 , 保存 到 Apache2.2\htdocs 目录 下 的 子 目录 4 中 , 然后 打开 正 
浏览 器 ， 在 地 址 栏 中 输入 http://localhost/4/4-2.php， 运 行 结果 如 下 所 示 : 


案例 4-2 
SStuNamel is an array:1l 
SStuName2 is an array:0 


4.2 ”管理 数组 


关于 管理 数组 的 内 容 ， 这 里 主要 介绍 数组 元 素 的 增加 及 删除 、 定 位 数组 元 素 以 及 确定 数组 的 大 
小 及 其 唯一 性 。 这 些 操 作 涉及 相当 多 的 数组 函数 ， 在 接 下 来 的 内 容 中 将 对 这 些 函数 进行 介绍 。 


4.2.1 增加 和 删除 数组 元 素 


在 实际 应 用 中 ， 数 组 元 素 的 增加 和 删除 是 常见 的 操作 。 这 两 种 操作 可 用 于 模仿 各 种 队列 的 实现 
(FIFO、LIFO 等 )。 其 中 ，FIFO 是 指 传统 的 队列 ， 它 是 一 种 数据 结构 ， 删 除 元 素 与 加 入 元 素 的 顺序 
相同 ， 所 以 称 为 先进 先 出 ; LIFO 是 指 栈 ， 它 是 另外 一 种 数据 结构 ， 删 除 元 素 的 顺序 与 加 入 时 的 顺序 
相反 ， 所 以 称 为 后 进 先 出 。 如 表 4-3 所 示 是 对 数组 元 素 增加 和 删除 所 涉及 的 函数 进行 的 介绍 。 


表 4-3 ”增加 和 删除 数组 元 素 

函数 说 明 
array_pushO 该 函数 将 variable 增加 至 target_array 的 末尾 ， 如 果 成 功 则 返回 True， 和 否则 返回 False。 向 该 函数 
传递 多 个 变量 作为 输入 参数 ， 同 时 向 数组 压 入 多 个 变量 元素) 
aray pop( | 该 函数 用 于 返回 target_array 的 最 后 一 个 元 素 ， 并 在 结束 后 重 置 煞 组 的 指针 
array_shift() 该 函数 类 似 于 array_pop0， 只 是 它 返 回 target_array 的 第 一 个 元 素 ， 而 非 最 后 一 个 。 其 结果 是 ， 
如 果 使 用 的 是 数值 键 , 则 所 有 相应 的 值 都 会 下 移 , 而 使 用 关联 键 的 数组 不 受 影响 。 与 array_popO 
一 样 ，array_shifti0 也 会 在 结束 时 重 置 指针 
array_unshift() | 该 函数 与 array_pushO 相 似 ， 只 是 它 将 元 素 增加 到 数组 头 ， 而 不 是 尾 。 所 有 已 有 的 数值 键 都 会 相 
应 地 修改 ， 反 映 出 在 数组 中 的 新 位 置 ， 而 关联 键 不 受 影 响 
array_pad0 该 函数 会 修改 target 数组 ， 将 其 大 小 增加 到 length 指定 的 长 度 。 这 是 通过 在 数组 中 填充 由 
pad _value 指定 的 值 实现 的 。 如 果 pad_value 是 正 数 ， 将 填充 到 数组 的 右 侧 (后 面 )， 如 果 为 负 ， 
将 填充 到 左 侧 (前 面 )。 如 果 length 等 于 或 小 于 当前 大 小 ， 将 不 做 任何 操作 
$arrayname[] ”| 确切 地 说 ， 它 不 是 一 个 函数 ， 而 是 一 个 语言 特性 


下 面 分 别 对 表 4-3 所 示 的 函数 的 格式 进行 介绍 及 举例 。 
(1) array pushO 的 使 用 格式 如 下 所 示 : 


int array _ push (array target array,mixed variable[,mixed variable...]) 


使 用 array_ pushO 函 数 的 示例 如 下 所 示 : 


$MyFriends = array(" 王 晶 晶 "," 李 晓 渊 "," 张 虎 ") ; 4 
array_push (SMYFriends," 汉 玉 杰 "," 龙 森 ") 7 “000® 
等 价 于 : 


$MyFriends = array(" 王 晶 晶 "," 李 晓 渊 "," 张 虎 "," 汉 玉 杰 "，" 龙 森 ") ; 
(2) array_pop0 的 使 用 格式 如 下 所 示 : 

mixed array top (array target array); 

使 用 array_pop0 函 数 的 示例 如 下 所 示 : 


array(" 王 晶 "," 李 晓 渊 "," 张 虎 ", " 冯 玉 杰 "，," 龙 菏 ") ; 
SoneFriend = array pop ($MyFriends); 


等 价 于 : 
$oneFriend = " 龙 生 "; 


(3) array_shift() 的 使 用 格式 如 下 所 示 : 
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$MyFriends 


mixed array shift(array target array) 


使 用 array_shift0 函 数 的 示例 如 下 所 示 : 


$MyFriends = array(" 王 晶 "," 李 晓 渊 "," 张 虎 "," 冯 玉 杰 "，," 龙 森 ") ; 
SoneFriend = array shift ($MyFriends); 
等 价 于 了 


SoneFriend = " 王 晶 "; 

(4) array_unshift0 函 数 的 示例 如 下 所 示 : 

int array unshift (array target array,mixed variablel[,mixed variable..]) 
使 用 array_unshift0 函 数 的 示例 如 下 所 示 : 


$MyFriends = array(" 王 晶 晶 "," 李 晓 渊 "," 张 虎 "," 冯 玉 杰 "," 龙 条 ") ; 
array_unshift ($MyFriends,，" 宋 岩 岩 "," 龙 建华 ") ; 


等 价 于 : 
$MyFriends = array(" 宋 岩 岩 ", " 龙 建华 ", " 王 晶 晶 "," 李 晓 渊 "," 张 虎 "," 冯 玉 杰 "," 龙 条") ; 
(5) array_pad0 的 使 用 格式 如 下 所 示 : 


array array pad(array target, integer length,mixed pad value) 


ih 


使 用 array_pad0 函 数 的 示例 如 下 所 示 : 


$MyFriends = array(" 李 晓 渊 "," 张 虎 "," 汉 玉 杰 "，" 龙 森 ") ; 
$MyFriends = array pad ($MyFriends,6€,"new Column"); 


等 价 于 : 
$MyFriends = array(" 李 晓 渊 "," 张 虎 "," 冯 玉 杰 "," 龙 森 ", "new Column", "new Column"); 


(6) $arrayname[] 可 以 用 类 似 于 如 下 所 示 的 赋值 方式 增加 数组 元 素 : 


$BoB["birthday"] = "1982-07-11"; 


卫 对 于 数值 索引 ， 可 以 使 用 如 下 所 示 的 方式 增加 一 个 新 元 素 : 
本 
去 $Friend[] = " 王 晶 晶 "7 
< 
DD 
届 几 ”4.2.2 定位 数组 元 素 
元 
全 
学 定位 数组 元 素 是 指 用 户 通过 一 些 函 数 有 效 地 筛选 数组 中 的 数据 ， 也 就 是 说 这 些 函 数 能 够 有 效 地 
于 定位 数组 中 的 元 素 ， 其 中 的 主要 函数 如 表 4-4 所 示 。 
册 
表 4-4 ”定位 数组 元 素 函 数 
函数 说 明 
in_array() 该 函数 用 于 检查 数组 中 是 否 存在 某 个 值 。 它 在 haystack 中 搜索 needle, 如 果 找到 则 返回 True， 
否则 返回 False。。 如 果 第 三 个 参数 strict 的 值 为 Tme， 则 in_array0 函 数 还 会 检查 needle 的 类 
型 是 否 和 haystack 中 的 相同 
array_keysO 该 函数 用 于 返回 数组 中 所 有 的 键 名 ， 也 就 是 返回 一 个 由 数组 target_array 中 所 有 键 组 成 的 数 
组 。 如果 指定 了 可 选 参数 search_value， 则 只 返回 该 值 的 键 名 。 否 则 target_array 数组 中 的 所 
有 键 名 都 会 被 返回 


array_keys_exists( | 该 函数 用 于 检查 给 定 的 键 名 或 索引 是 否 存在 于 数组 中 。 它 所 给 定 的 key 存在 于 数组 中 时 返回 
True。key 可 以 是 任何 能 作为 数组 索引 的 值 
array_values() 该 函数 用 于 返回 target_array 数组 中 所 有 的 值 并 给 其 建立 数字 索引 
array search() 该 函数 用 于 在 数组 中 搜索 给 定 的 值 ， 如 果 成 功 则 返回 相应 的 键 名 


下 面 分 别 对 表 4-4 所 示 函 数 的 格式 进行 介绍 及 举例 。 
(1) in_array0 的 使 用 格式 如 下 所 示 : 


bool in array (mixed needle, array haystack [, bool strict]) 
使 用 in_array0 函 数 的 示例 如 下 所 示 : 


S$ages = array(16,17,18,19,20,21,22,23,24,25,26); 
if (in array("25",$ages)) 

{echo "Search 25 Success!<br>";} 

if (in array(25, $ages,1)) 


上 述 代码 成 功 执行 后 ， 将 输出 如 下 信息 : 


从 执行 结果 可 以 看 出 ， 第 二 个 测试 要 求 数据 类 型 必须 匹配 。 由 于 第 二 个 测试 完成 的 是 整数 和 整 
数 的 比较 ， 所 以 测试 成 功 ， 如 果 它 们 类 型 不 同 将 导致 失败 。 
(2) array_keys0 的 使 用 格式 如 下 所 示 : 


使 用 array_keys0 函 数 的 示例 如 下 所 示 : 


上 述 代码 成 功 执行 后 ， 将 输出 如 下 信息 : 


(3) array_keys_exists0 的 使 用 格式 如 下 所 示 : 


使 用 array_keys_exists0 函 数 的 示例 如 下 所 示 : 


上 述 代 码 成 功 执行 后 ， 将 输出 如 下 信息 : 
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The "first” element is in the array 
(4) array_values0 的 使 用 格式 如 下 所 示 : 
array array values(array target array) 
使 用 array_values0 函 数 的 示例 如 下 所 示 : 


$array = array ("size" => "XL", "color" => "red"); 
print rl(array values ($array)); 


上 述 代码 成 功 执行 后 ， 将 输出 如 下 信息 : 


( 
[0] => XI 
[1] => red 
有 


(5) array_search() 的 使 用 格式 如 下 所 示 : 
mixed array search ( mixed needle, array haystack [, bool strict] ) 


该 函数 在 haystack 中 搜索 needle 参数 ， 如 果 找 到 则 返回 键 名 ， 和 否则 返回 False。 同 前 面 一 样 ， 如 
果 可 选 的 第 三 个 参数 strict 为 Tme， 则 array_search0 函 数 还 将 在 haystack 中 检查 needle 的 类 型 。 


4.2.3 ”确定 数组 大 小 和 唯一 性 


在 PHP 中 ,有 些 函数 可 以 用 来 确定 数组 中 值 的 总 数 以 及 唯一 值 的 个 数 。 接 下 来 将 对 这 些 函 数 进 
行 介绍 ， 如 表 4-5 所 示 。 


表 4-5 确定 数组 大 小 和 唯一 性 的 函数 


该 函数 用 于 返回 input_array 中 元 素 的 总 数 。 如 果 启 用 了 可 选 的 mode 参数 〈 设 置 为 1)， 数 
组 将 进行 递归 计数 。 在 统计 多 维 数组 中 所 有 元 素 的 个 数 时 这 个 特性 很 有 用 
该 函数 用 于 统计 数组 中 所 有 的 值 出 现 的 次 数 。 统 计 完成 后 返回 一 个 包含 关联 键 / 值 对 的 数 
组 。 其 中 每 个 键 表示 input_array 中 的 一 个 值 ， 相 应 的 值 表示 这 个 键 在 input_array 中 出 现 的 
频 度 

该 函数 将 会 删除 input_array 中 所 有 重复 的 值 ,返回 一 个 由 唯一 值 组 成 的 数组 array_uniqueO 
函数 先 将 值 作为 字符 串 排序 ， 然 后 对 每 个 值 只 保留 第 一 个 遇 到 的 键 名 ， 接 着 忽略 所 有 后 面 
的 键 名 。 这 并 不 意味 着 在 未 排序 的 array 中 同一 个 值 的 第 一 个 出 现 的 键 名 会 被 保留 


array_count_ valuesO) 


array_unique() 


下 面 分 别 对 表 4-5 所 示 函 数 的 格式 进行 介绍 及 举例 : 
1) count0 的 使 用 格式 如 下 所 示 : 


int count (array input array [, int mode] ) 


使 用 countO 函 数 的 示例 如 下 所 示 : 


一 


执行 上 述 语句 代码 将 输出 信息 : 5。 
countO 函 数 的 另 一 种 示例 如 下 所 示 : 


上 述 语句 代码 统计 了 StuName 中 的 标量 元 素 个 数 和 数组 个 数 。 它 将 输出 信息 : 6。 

虽然 语句 中 看 起 来 数组 中 只 有 5 个 元 素 ， 但 这 里 有 一 个 保存 “ 王 晶 晶 ””“ 赵 星 ” 和 “ 李 晓 渊 ” 
的 数组 实体 。 所 以 ， 不 但 将 这 个 数组 的 内 容 “ 王 晶 晶 ”“ 赵 星 ” 和 “ 李 晓 渊 ”) 分 别 统计 为 一 个 元 
素 ， 而 且 还 将 这 个 数组 实体 本 身 也 统计 为 一 个 元 素 。 

(2) array_count valuesO 的 使 用 格式 如 下 所 示 : 


使 用 array_count_valuesO 函 数 的 示例 如 下 所 示 : 


上 述 代 码 成 功 执行 后 ， 将 输出 如 下 信息 : 


(3) array_uniqueO 的 使 用 格式 如 下 所 示 : 


使 用 array_uniqueO 函 数 的 示例 如 下 所 示 : 


上 述 代码 成 功 执行 后 ， 将 输出 如 下 信息 : 
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数据 应 用 是 指 对 数组 的 常用 操作 ， 比 较 常见 的 有 : 遍历 数组 、 数 组 排序 、 合 并 、 拆 分 、 接 合 、 
分 解数 组 等 。 


4.3.1 ”遍历 数组 


遍历 数组 是 指 访问 数组 中 的 每 个 元 素 ， 直 到 没有 没 被 访问 的 元 素 为 止 。 在 PHP 中 ,遍历 数组 既 
包括 遍历 顺序 数组 ， 也 包括 遍历 非 顺序 数组 〈 标 识 符 全 为 整数 时 不 连续 或 标识 符 不 全 为 整数 时 ) 。 
下 面 将 分 别 进行 介绍 。 

1. 遍历 顺序 数组 

在 数组 中 ， 标 识 符 是 连续 整数 的 数组 被 称 为 顺序 数组 。 顺 序数 组 中 最 小 的 标识 符 值 通常 为 0， 
但 用 户 可 以 创建 一 个 最 小 标识 符 值 是 1 或 其 他 任何 整数 的 顺序 数组 。 如 果 知 道 一 个 顺序 数组 的 最 小 
标识 符 值 ， 则 可 以 使 用 for 循环 语句 遍历 整个 数组 。 要 遍历 整个 数组 可 以 将 循环 变量 初始 化 为 最 小 
标识 符 值 , 然后 使 用 count0 函 数 形成 循环 测试 表达 式 。 这 里 count0 函 数 用 于 返回 数组 中 的 元 素数 量 。 
下 面 的 代码 用 于 遍历 一 个 数组 : 


<?php 
$stuName = array (0=>" 打 岩 岩 ", 1=>" 龙 建华 "，2 => " 王 晶 晶 ",3 => " 赵 星 ", 4 =>" 李 晓 渊 ") ; 
$sum = count ($stuName); 
for ($i=0;$i<$sum; $i++) 
{ 
echo "<br>$i => $sStuName [$i]"; 
} 


?> 


在 上 述 代码 中 ， 第 一 行 语句 用 于 创建 数组 ， 并 进行 初始 化 。 第 二 行 语句 用 于 获得 数组 中 的 元 素 
数量 。 接 下 来 的 for 语句 使 用 循环 变量 i 在 数组 内 遍历 ， 它 的 主体 使 用 echo 输出 各 数组 的 标识 符 和 
值 。 具 体 如 下 所 示 : 


=> 朱 岩 岩 
=> 龙 建华 
=> 王 晶 晶 
=> 赵 星 

=> 李 晓 渊 


2. 遍历 非 顺 序数 组 
在 PHP 中 要 遍历 非 顺序 数组 要 使 用 foreach 语句 ， 它 的 使 用 使 遍历 非 顺序 数组 变 得 非常 简单 。 
foreach 语句 的 格式 如 下 所 示 : 


wb 


foreach ($array as $key => $value) { body } 


上 述 语句 用 于 在 名 为 array 的 数组 内 进行 遍历 ， 相 应 地 为 每 个 数组 元 素 设置 key 和 value 的 值 。 
它 的 使 用 如 下 所 示 : 


<?php : 
SStuName = array ("BoB"”=> " 宋 岩 岩 "， "Jack"” => " 龙 建 华 ",2 => " 王 晶 晶 ") ; 局 
foreach ($StuName as $index => $StuN) 9 


{ 
echo "<br>$index => $StuN"; 
} 


2> 


在 上 述 语 句 中 , echo 语句 只 引用 变量 index 和 StuN 的 值 , 它们 被 自动 设置 为 当前 标识 符 和 元 素 


的 值 。 它 的 输出 如 下 所 示 : 2 
BoB => 宋 岩 岩 二 
Jack => 龙 建 华 浊 
2 => 王 晶 晶 
3. 遍历 数组 函数 
遍历 数组 所 涉及 的 函数 很 多 ， 这 里 只 介绍 一 些 在 遍历 数组 时 经 常用 到 的 函数 ， 具 体 如 表 4-6 

所 示 。 

表 4-6 遍历 数组 函数 
函数 说 明 

key0 该 函数 返回 数组 中 位 于 当前 指针 位 置 的 键 元 素 。 这 里 需要 注意 ， 每 个 key0 函 数 调用 不 会 推进 指 
针 。 为 此 要 使 用 next0 函 数 ， 这 个 函数 的 唯一 作用 就 是 完成 推进 指针 的 任务 

reset() 该 函数 用 来 将 数组 的 指针 设置 回 数组 的 开始 位 置 。 如 果 需 要 在 一 个 脚本 中 多 次 查看 或 处 理 一 个 
数组 ， 就 会 经 常 使 用 这 个 函数 ， 另 外 ， 这 个 函数 还 常用 于 排序 结束 时 

eachO 该 函数 返回 数组 的 当前 键 / 值 对 ， 并 将 指针 推进 一 个 位 置 。 返 回 的 数组 包含 4 个 键 ， 键 0 和 key 
包含 键 名 ， 而 键 1 和 value 包含 相应 的 数据 。 如 果 执 行 eachO 函 数 前 指针 位 于 数组 末尾 ， 则 返回 
False 

current() 该 函数 返回 位 于 数组 当前 指针 位 置 的 数组 值 。 它 与 next0、prev0 和 end0 函 数 不 同 ，current0 函 
数 不 移 动 指针 

endO 该 函数 将 指针 移 向 数组 的 最 后 一 个 位 置 ， 并 返回 最 后 一 个 元 素 

next() 该 函数 返回 紧 接 着 放 在 当前 数组 指针 的 下 一 个 位 置 的 数组 值 

prevO 该 函数 返回 位 于 当前 指针 前 一 个 位 置 的 数组 值 ， 如 果 指 针 本 来 就 位 于 数组 的 第 一 个 位 置 ， 则 返 


加 False 

array_walkO ”| 该 函数 将 数组 中 的 各 个 元 素 传递 给 自 定义 函数 function。 当 用 户 需要 对 各 个 数组 元 素 完成 某 个 特 
定 动作 时 ， 这 个 函数 就 很 有 用 。 如 果 希 望 真正 修改 数组 键 / 值 对 ， 就 需要 将 每 个 键 / 值 对 作为 引用 
传递 给 函数 。 这 里 需要 注意 ， 自 定义 函数 必须 接收 两 个 输入 参数 :第 一 个 表示 数组 的 当前 值 ， 
第 二 个 表示 当前 键 。 如 果 调 用 array_walkO 函 数 时 给 出 了 可 选 的 userdata 参数 ， 它 的 值 将 作为 第 
三 个 参数 传递 给 自 定义 函数 
array_reverse() | 该 函数 将 数组 中 元 素 的 顺序 逆 置 。 如 果 其 可 选 参数 为 True， 则 保持 键 映 射 。 和 否则 ， 重 新 摆 放 后 
的 各 个 值 将 对 应 于 先前 该 位 置 上 的 相应 键 
array flipO) 该 函数 将 使 数组 中 键 及 其 相应 值 倒 换 角色 


4.3.2 ”数组 排序 


数组 排序 是 在 数组 上 执行 的 另 一 个 常见 操作 。PHP 提供 了 一 组 函数 ， 如 表 4-7 所 示 。 通 过 使 用 
这 些 函 数 ， 使 对 数组 排序 变 得 非常 简单 。 在 默认 情况 下 ，PHP 的 排序 函数 按 英 语 指定 的 规则 进行 反 
序 。 如 果 需 要 按 另 一 种 语言 的 约定 进行 排序 ， 比 如 中 文 、 法 语 或 德语 ， 就 需要 修改 此 默认 行为 ， 可 
以 使 用 setlocale0 函 数 设 置 本 地 化 环境 (locale)。 


表 4-7 数组 排序 函数 


函数 说 了 明 
卫 sortO 该 函数 用 于 对 target_array 进行 排序 ， 各 元 素 按 值 由 低 到 高 的 顺序 排列 。 这 里 需要 注意 ， 它 并 
站 不 返回 排序 后 的 数组 。 相 反 ， 它 只 是 对 数组 排序 ， 不 论 结果 如 何 都 不 返回 任何 值 
3 natsortO) 该 函数 实现 了 一 个 和 人 们 通常 对 字母 数字 字符 串 进行 排序 的 方法 一 样 的 排序 算法 ， 并 保持 原 
2 有 键 / 值 的 关联 ， 这 被 称 为 “自然 排序 ” 
完 natcasesort() 该 函数 用 于 “自然 排序 ”算法 对 数组 进行 不 区 分 大 小 写字 母 的 排序 , 也 就 是 说 函数 natcasesort() 
宇 在 功能 上 与 natsort0 函 数 相 同 ， 只 是 它 不 区 分 大 小 写 
习 rsortO 该 函数 与 sortO 函 数 相同 ， 只 是 它 以 相反 的 顺序 〈 降 序 ) 对 数组 元 素 排序 
于 asort() 该 函数 与 sort0 函 数 相同 ， 对 数组 进行 排序 ， 数 组 的 索引 保持 和 单元 的 关联 。 主 要 用 于 对 那些 


单元 顺序 很 重要 的 结合 数组 进行 排序 
array_multisort( | 该 函数 可 以 用 来 一 次 对 多 个 数组 进行 排序 ， 或 者 根据 某 一 维 或 多 维 对 多 维 数组 进行 排序 。 关 
联 (string) 键 名 保持 不 变 ， 但 数字 键 名 会 被 重新 索引 。 如 果 成 功 则 返回 Te， 否则 返回 False 


arsort() 该 函数 与 asort0 函 数 一 样 ， 会 保持 键 / 值 对 的 关联 。 但 是 ， 它 以 逆序 对 数组 排序 。 如 果 包 括 了 
可 选 的 sort_flags 参数 ， 具 体 的 排序 行为 将 由 这 个 值 来 确定 

ksort() 该 函数 按键 对 输入 数组 array 排序 ， 如 果 成 功 则 返回 Trme， 否 则 返回 False。 如 果 包 括 了 可 选 
的 sort_flags 参数 ， 具 体 的 排序 行为 将 由 这 个 值 来 确定 

krsortO 该 函数 与 ksort0 函 数 相同 ， 也 会 按键 排序 ， 只 是 它 会 以 逆序 降序) 排列 

usortO 该 函数 将 用 用 户 自 定义 的 比较 函数 对 一 个 数组 中 的 值 进行 排序 。 如 果 要 排序 的 数组 需要 用 一 


种 不 寻常 的 标准 进行 排序 ， 那 么 应 该 使 用 此 函数 。 用 户 自 定义 函数 必须 接收 两 个 输入 参数 ， 
根据 第 一 个 参数 小 于 、 等 于 或 大 于 第 二 个 参数 , 相应 地 返回 一 个 负 整 数 、0 或 正 整数 。 很 显然 ， 
这 个 函数 必须 与 usort0 函 数 调用 在 相同 的 作用 域内 


下 面 分 别 对 表 4-7 所 示 函 数 的 格式 进行 介绍 及 举例 。 
1. sort() 
它 的 使 用 格式 如 下 所 示 : 


void sort ( array target array[,int $sort flags]) 


在 上 述 格式 中 ，sort_flags 参数 可 选 ， 将 根据 这 个 参数 指定 的 值 修改 该 函数 的 默认 行为 。 

。 SORIT_ NUMBERIC 按 数 值 排序 。 对 整数 或 浮 点 数 排序 时 很 有 用 。 

。 SORT_REGULAR 按照 相应 的 ASCII 值 对 元 素 排 序 。 例 如 ,也 的 ASCI 值 在 a 的 前 面 。 
。 SORT_STRING 按 接 近 于 人 所 认 知 的 正确 顺序 对 元 素 排 序 。 

使 用 sort0 函 数 的 示例 如 下 所 示 : 


上 述 代码 成 功 执行 后 ， 将 输出 如 下 信息 : 


2. natsort() 
它 的 使 用 格式 如 下 所 示 : 


使 用 natsortO 函 数 的 示例 如 下 所 示 : 


上 述 代码 成 功 执行 后 ， 将 输出 如 下 信息 : 


3. natcasesort() 


它 的 使 用 格式 如 下 所 示 : 

voidnatcasesort (array target array) 
卫 4. rsort() 
王 它 的 使 用 格式 如 下 所 示 : 
马 | voida rsort(array target array [iint sort flags] ) 
名 
完 在 上 述 格式 中 ， 如 果 包括 了 可 选 的 sort_flags 参数 ， 那 么 具体 的 排序 行为 将 由 这 个 值 来 确定 。 
金 5. asort() 
习 它 的 使 用 格式 如 下 所 示 : 
手 
册 


使 用 asort0 函 数 的 示例 如 下 所 示 : 


上 述 代 码 成 功 执行 后 ， 将 输出 如 下 信息 : 


6. array_multisort() 
它 的 使 用 格式 如 下 所 示 : 


该 函数 的 参数 结构 非常 灵活 。 第 一 个 参数 必须 是 一 个 数组 。 接 下 来 的 每 个 参数 可 以 是 数组 或 者 
是 下 面 列 出 的 排序 标志 。 

(1) 排序 顺序 标志 

。 SORT_ASC 按照 上 升 顺序 排序 。 

。 SORT_DESC 按照 下 降 顺 序 排序 。 


(2) 排序 类 型 标志 

。 SORT_REGULAR 将 项 目 按照 通常 方法 比较 。 

。 SORT_NUMERIC 将 项 目 按照 数值 比较 。 

。 SORT_STRING 将 项 目 按照 字符 串 比 较 。 

每 个 数组 之 后 不 能 指定 两 个 同类 的 排序 标志 。 每 个 数组 后 指定 的 排序 标志 仅 对 该 数组 有 效 ， 在 
此 之 前 为 默认 值 SORT_ ASC 和 SORT_ REGULAR。 使 用 array_multisort0 函 数 的 示例 如 下 所 示 : 


上 述 代 码 成 功 执行 后 ， 将 输出 如 下 信息 : 


7. arsort() 
它 的 使 用 格式 如 下 所 示 : 


8. ksort() 
它 的 使 用 格式 如 下 所 示 : 


9. krsort() 
它 的 使 用 格式 如 下 所 示 : 


10. usort() 
它 的 使 用 格式 如 下 所 示 : 
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使 用 usort0 函 数 的 示例 如 下 所 示 : 


上 述 代 码 成 功 执行 后 ， 将 输出 如 下 信息 : 


4.3.3 合并 、 拆 分 、 接 合 和 分 解数 组 


PHP 提供 了 一 些 函 数 来 完成 相对 比较 复杂 的 数组 处 理 任务 ， 如 数组 的 合并 、 拆 分 、 接 合 及 分 解 
等 。 下 面 将 分 别 介绍 与 此 相关 的 函数 。 
1. array_combine() 


它 的 使 用 格式 如 下 所 示 : 


该 函数 用 于 创建 一 个 数组 ， 用 一 个 数组 的 值 作为 其 键 名 ， 另 一 个 数组 的 值 作为 其 值 。 在 使 用 格 
式 中 它 由 输入 参数 数组 keys 中 的 键 和 输入 参数 数组 values 中 对 应 的 值 组 成 。 这 里 需要 注意 的 是 , 两 
个 输入 数组 的 大 小 必须 相同 ， 不 能 为 室 。array_combineO 函 数 的 具体 示例 如 下 所 示 : 


上 述 代码 成 功 执行 后 ， 将 输出 如 下 信息 : 


Array 

( 
[red] => apple 
[yellow] => banana 


Hl 


2. array_merge() 


它 的 使 用 格式 如 下 所 示 : 
array array merge (array input arrayl[, array input array2 [...,array input arrayN]]) 


该 函数 用 于 合并 一 个 或 多 个 数组 ， 也 就 是 将 一 个 或 多 个 数组 的 单元 合并 起 来 ， 一 个 数组 中 的 值 
附加 在 前 一 个 数组 的 后 面 。 返 回 作为 结果 的 数组 。 如 果 输 入 的 数组 中 有 相同 的 字符 串 键 名 ， 则 该 键 
名 后 面 的 值 将 覆盖 前 一 个 值 。 然 而 ， 如 果 数 组 包含 数字 键 名 ， 后 面 的 值 将 不 会 覆盖 原来 的 值 ， 而 是 
附加 到 后 面 。 如 果 只 给 出 一 个 数组 并 且 该 数组 是 数字 索引 的 ， 则 键 名 会 以 连续 方式 重新 索引 。 
array_merge() 函 数 的 具体 示例 如 下 所 示 


S$arrayl = array ("color" => "red", 2, 4); 

$array2 = array("a", "b", "color" => "green", "shape" => "circle", 4); 
$result = array merge ($arrayl, $array2); 

print r($result); 


上 述 代 码 成 功 执行 后 ， 将 输出 如 下 信息 : 


Array 
《 


[color] => green 


[0] => 2 
[1] => 4 
[2] => a 
[3] => b 
[shape] => circle 
[4] => 4 


3. array_merge_recursive() 
它 的 使 用 格式 如 下 所 示 : 


array array merge recursive ( array input arrayl , array input array2 [,array] ) 


该 函数 用 于 将 一 个 或 多 个 数组 的 单元 合并 起 来 ， 一 个 数组 中 的 值 附加 在 前 一 个 数组 的 后 面 。 返 
回 作为 结果 的 数组 。 它 与 array_merge0 函 数 的 区 别 在 于 ， 当 某 个 输入 数组 中 的 某 个 键 已 经 存在 于 结 
果 数 组 中 时 处 理 方式 不 同 。array_mergeO 函 数 会 覆盖 前 面 存在 的 键 / 值 对 ， 蔡 换 为 当前 输入 数组 中 的 
键 / 值 对 ， 而 array_merge_recursive0 函 数 将 把 两 个 值 合并 在 一 起 ， 形 成 一 个 新 的 数组 ， 并 以 现 有 的 键 
作为 数组 名 。array_merge _Iecursive0 函 数 的 具体 示例 如 下 所 示 : 


Sarl = array("color" => "red”", 5); 
$ar2 = array(10, "color" => "green", "blue"); 
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上 述 代码 成 功 执行 后 ， 将 输出 如 下 信息 : 


4. array_slice() 
它 的 使 用 格式 如 下 所 示 : 


该 函数 用 于 返回 根据 offset 和 length 参数 所 指定 的 input_array 数组 中 的 一 段 序列 。 如 果 offset 
非 负 ， 则 序列 将 从 input_array 中 的 此 偏 移 量 开 始 。 如 果 offset 为 负 ， 则 序列 将 从 input_array 中 距离 
末端 为 此 偏 移 量 长 度 的 位 置 开始 。 

如 果 给 出 了 length 并 且 为 正 , 则 会 在 距离 数组 开头 的 offsettength 位 置 结束 。 如 果 给 出 了 length 
并 且 为 负 ， 则 在 距离 数组 开头 的 count(input_array)-|length| 位 置 结束 。 如 果 省 略 ， 则 序列 将 从 offset 
开始 一 直到 input_array 的 末端 。array_slice0 函 数 的 具体 示例 如 下 所 示 : 


上 述 代 码 成 功 执行 后 ， 将 输出 如 下 信息 : 


5. array_splice() 


它 的 使 用 格式 如 下 所 示 : 
array array splice ( array input, int offset [, int length [, array replacenent ]] ) 


该 函数 用 于 把 input 数组 中 由 offset 和 length 指定 的 单元 去 掉 ， 如 果 提 供 了 replacement 参数 ， 
则 用 replacement 数组 中 的 单元 取代 。 返 回 一 个 包含 被 移 除 单元 的 数组 。 这 里 需要 注意 的 是 input 中 
的 数字 键 名 不 被 保留 。array_splice0 函 数 的 具体 示例 如 下 所 示 : 


$input = array("red", "green", "blue", "yellow"); 
$subl = array splice ($input, 2); 0 
print r($subl); 

$input = array("red", "green", "blue", "yellow"); 

$sub2 = array splice($input, 1, -1); 

print r($sub2); 


上 述 代 码 成 功 执行 后 ， 将 输出 如 下 信息 : 


Array([0] => blue [1] => yellow) 
Array([0] => green [1] => blue) 
6. array_intersect() 

它 的 使 用 格式 如 下 所 示 : 


array array intersect (array arrayl, array array2 [, array...]) 


该 函数 用 于 计算 数组 的 交集 。 它 返回 一 个 数组 ， 该 数组 包含 了 所 有 在 arrayl 中 同时 也 出 现在 所 
有 其 他 参数 数组 中 的 值 。array_intersect0 函 数 的 具体 示例 如 下 所 示 : 


苘 洪山 太 潍 


$arrayl = array("a" => "green", "red", "blue"); 
$array2 = array("b" => "green", "yellow", "red"); 
$result = array intersect ($arrayl, $array2); 
print r($result); 


上 述 代 码 成 功 执行 后 ， 将 输出 如 下 信息 : 


Array 
[a] => green 
[0] => red 
) 
7. array_intersect_assoc() 


它 的 使 用 格式 如 下 所 示 : 


array array_intersect_assoc( array arrayl, array array2 [, array...] ) 


该 函数 与 array_intersectO 函 数 基本 相同 ， 只 不 过 它 在 比较 中 还 考虑 了 数组 的 键 。 因 此 ， 只 有 在 
arrayl 中 出 现 , 且 在 所 有 其 他 输入 数组 中 也 出 现 的 键 / 值 对 才 返 回 到 结果 数组 中 。array_intersect_assoc() 
函数 的 具体 示例 如 下 所 示 : 


Sarrayl = array("a" => "green", "b" => "brown", "c" => "blue", "red"); 


Sarray2 = array("a" => "green", "yellow", "red"); 
$result array = array intersect assoc($arrayl, $array2); 


print rl($result array); 
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上 述 代码 成 功 执行 后 ， 将 输出 如 下 信息 : 


8. array_diff() 
它 的 使 用 格式 如 下 所 示 : 


该 函数 用 于 计算 数组 的 差 集 。 它 返回 一 个 数组 ， 该 数组 包括 了 所 有 在 array1 中 但 不 在 任何 其 他 
参数 数组 中 的 值 。 数 组 的 键 名 保留 不 变 。array_diff0 函 数 的 具体 示例 如 下 所 示 : 


上 述 代 码 成 功 执行 后 ， 将 输出 如 下 信息 : 


9. array_diff_assoc() 
它 的 使 用 格式 如 下 所 示 : 


该 函数 与 aray_diff0 函 数 基本 相同 ， 只 是 它 在 比较 时 还 考虑 了 数组 的 键 。 因 此 ， 只 在 array1 中 
出 现 而 不 在 其 他 输入 数组 中 出 现 的 键 / 值 对 才 会 返回 到 结果 数组 中 。array_di 全 assoc0 函 数 的 具体 示 
例如 下 所 示 : 


上 述 代码 成 功 执行 后 ， 将 输出 如 下 信息 : 


4.3.4 其 他 数组 函数 


还 有 一 些 关 于 数组 的 函数 前 面 并 没有 介绍 , 但 它们 也 经 常用 到 , 本 小 节 就 介绍 一 些 这 样 的 函数 。 
1. array_rand() 
它 的 使 用 格式 如 下 所 示 : 


mixed array rand ( array input array [, int num entries] ) 


该 函数 将 返回 input_array 中 的 一 个 或 多 个 键 。 如 果 忽 略 可 选 的 num_entries 参数 ， 则 只 返回 一 
个 随机 值 。 可 以 通过 设置 num_entries 来 调整 所 返回 随机 值 的 个 数 。 


如 果 只 取出 一 个 ，array_randO 函 数 则 返回 一 个 随机 单元 的 键 名 ， 和 否则 返回 一 个 包含 随机 键 名 的 % 
数组 。 这 样 可 以 随机 从 数组 中 取出 键 名 和 值 。array_rand0 函 数 的 具体 示例 如 下 所 示 : 章 
$input = array(" 宋 岩 岩 "，" 龙 建华 "，" 王 晶 "，" 赵 星 "，" 李 晓 渊 ") ; 出 


Srand keys = array rand($input, 2); 
print "rand keys[0]: ".$input[$rand keys[0]] . "\n"; 
print "rand keys[1]: ".$input[$rand keys[1]] . "\n"; 


上 述 代 码 成 功 执行 后 ， 将 输出 类 似 于 如 下 信息 : 


rand keys[0]: 宋 岩 岩 
rand keys [1]: 龙 建 华 


2. array_sum() 

它 的 使 用 格式 如 下 所 示 : 

mixed array sum ( array input array ) 

该 函数 将 input_array 中 的 所 有 值 加 在 一 起 ， 返 回 最 终 的 和 。 其 中 ， 所 有 值 都 必须 是 整数 或 浮 点 
数 。 如 果 数 组 中 包含 其 他 数据 类 型 〈 例 如 字符 串 )， 这 些 值 将 被 忽略 。array_sum0 函 数 的 具体 示例 如 
下 所 示 : 

Sa = erayl2, A 6 8) 

echo "sum(a) = " . array sum($a) . "\n™; 


Sb = array("a” => 7.11, "b™ => 6.24; "Cc" 一 > 9.82) 2 
echo "sum(b) = " . array sum($b) . "\n™; 


上 述 代 码 成 功 执行 后 ， 将 输出 如 下 信息 : 


sum(a) = 20 
sum(b) = 33.17 


3. array_chunk() 
它 的 使 用 格式 如 下 所 示 : 


array array chunk ( array input array, int size [, boolean preserve keys] 
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该 函数 将 一 个 数组 分 割 成 多 个 数组 ， 其 中 每 个 数组 的 单元 数目 由 size 决定 。 最 后 一 个 数组 的 单 
元 数目 可 能 会 少 于 size。 最 后 得 到 的 数组 是 一 个 多 维 数组 中 的 单元 ， 其 索引 从 零 开始 。 但 将 可 选 参 
数 preserve keys 设 为 Tue， 可 以 使 PHP 保留 输入 数组 中 原来 的 键 名 。 如 果 用 户 指定 了 False， 那 么 
每 个 结果 数组 将 用 从 零 开始 的 新 数字 索引 。 默 认 值 是 False。 array_chunkO 函 数 的 具体 示例 如 下 所 示 : 


上 述 代码 成 功 执行 后 ， 将 输出 如 下 信息 : 


4. array_fill() 
它 的 使 用 格式 如 下 所 示 : 


该 函数 用 给 定 的 值 填充 数组 。 它 用 value 参数 的 值 将 一 个 数组 填充 num 个 条 目 ， 键 名 的 开始 由 
start_index 参数 指定 。 这 里 需要 注意 的 是 ，num 必须 是 一 个 大 于 零 的 数值 ， 否 则 PHP 会 发 出 一 条 警 
告 。array_fil0 函 数 的 具体 示例 如 下 所 示 : 


上 述 代 码 成 功 执行 后 ， 将 输出 如 下 信息 : 


5. array_filter() 
它 的 使 用 格式 如 下 所 示 : 


该 函数 用 回调 函数 过 滤 数 组 中 的 单元 ， 它 依次 将 input_array 数组 中 的 每 个 值 传递 到 callback 函 
数 。 如 果 callback 函数 返回 Trme， 则 input_array 数组 的 当前 值 会 被 包含 在 返回 的 结果 数组 中 。 数 组 
的 键 名 保留 不 变 。array_filter0 函 数 的 具体 示例 如 下 所 示 : 


Ln 


类 
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组 


上 述 代 码 成 功 执行 后 ， 将 输出 如 下 信息 : 


6. array_search() 


它 的 使 用 格式 如 下 所 示 : 
‘mixed array search (mixed needle, array haystack [, bool strict]) | 


该 函数 用 于 在 数组 中 搜索 给 定 的 值 ， 如 果 成 功 则 返回 相应 的 键 名 ， 即 在 haystack 中 搜索 needle 
参数 ， 并 在 找到 的 情况 下 返回 键 名 ， 否 则 返回 False。array_searchO 函 数 的 具体 示例 如 下 所 示 : 
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上 述 代码 成 功 执行 后 ， 将 输出 如 下 信息 : 


7. shuffle() 
它 的 使 用 格式 如 下 所 示 : 


该 函数 用 于 打 乱 〈 随 机 排列 单元 的 顺序 ) 一 个 数组 ， 也 就 是 随机 地 对 input_array 中 的 元 素 重新 排 
序 。 需 要 注意 的 是 ， 该 函数 为 array 中 的 单元 赋予 新 的 键 名 ， 这 将 删除 原 有 的 键 名 而 不 仅 是 重新 排 
序 。shuflle0 函 数 的 具体 示例 如 下 所 示 : 


上 述 代 码 成 功 执行 后 ， 将 输出 类 似 于 如 下 所 示 的 信息 : 


8. list() 
它 的 使 用 格式 如 下 所 示 : 


该 函数 用 于 在 一 个 单独 的 赋值 语句 中 向 多 个 变量 赋值 。 因 为 list0 函 数 使 访问 eachO 函 数 返 回 的 
键 和 值 变 得 非常 容易 ， 所 以 list0 函 数 经 常 与 eachO 函 数 一 起 使 用 。 使 用 格式 中 说 明 从 varl 到 varN 
的 第 一 个 指定 变量 都 将 从 数组 array_value 接收 到 一 个 值 。 在 某 种 意义 上 ，list0 函 数 与 arrayO 函 数 相 
反 ， 因 为 listO 函 数 将 一 个 数组 划分 为 一 个 系统 标量 值 ， 而 array0 函 数 根据 一 个 系统 标量 值 构 造 一 个 
数组 。list0 函 数 的 具体 示例 如 下 所 示 : 


上 述 代 码 成 功 执行 后 ， 将 输出 类 似 于 如 下 所 示 的 信息 : 


4.4 PHP 和 HTML 表单 


PHP 和 HTML 有 很 多 相互 作用 : PHP 能 生成 HTML，HTML 可 以 向 PHP 程序 传递 信息 。 通 常 
情况 下 , 多 数 PHP 程序 使 用 HTML 表单 获得 用 户 输入 的 数据 。 本 节 将 介绍 与 HTML 表单 相关 的 Get 
和 Post， 以 及 如 何 获取 表单 提交 的 数据 等 方面 的 内 容 。 


4.4.1 HTML 表单 Get 和 Post 


在 B/S 应 用 程序 中 ， 前 台 与 后 台 的 数据 交互 ， 都 是 通过 HTML 中 的 Form 表单 完成 的 。 Form 提 
供 了 两 种 数据 传输 的 方式 一 一 Get 和 Post。 虽 然 它 们 都 是 数据 的 提交 方式 ， 但 是 在 实际 传输 时 却 有 
很 大 的 不 同 ， 并 且 可 能 会 对 数据 产生 严重 的 影响 。 为 了 使 程序 开发 者 更 加 方便 地 得 到 变量 值 ，Web 
容器 已 经 屏蔽 了 二 者 的 一 些 差异 ， 但 是 了 解 二 者 的 差异 在 以 后 的 编程 中 会 很 有 帮助 。 

Form 中 的 Get 和 Post 方法 ， 在 数据 传输 过 程 中 分 别 对 应 HTTP 协议 中 的 Get 和 Post 方法 。 二 
者 的 主要 区 别 如 下 : 

(1) Get 用 来 从 服务 器 上 获得 数据 ， 而 Post 用 来 向 服务 器 上 传 数据 。 

(2) Get 将 表单 中 的 数据 按照 variable=value 的 形式 ， 添 加 到 action 所 指向 的 URL 后 面 ， 并 且 
两 者 使 用 “?” 连 接 ， 而 各 个 变量 之 间 使 用 “&” 连 接 ; Post 是 将 表单 中 的 数据 放 在 Form 的 数据 体 
中 ， 按 照 变 量 和 值 相对 应 的 方式 ， 传 递 到 action 所 指向 的 URL。 

(3) Get 是 不 安全 的 ， 因 为 在 传输 过 程 中 ， 数 据 被 放 在 请 求 的 URL 中 ， 而 如 今 现 有 的 很 多 服务 
器 、 代 理 服务 器 或 者 用 户 代理 都 会 将 请 求 URL 记录 到 日 志文 件 中 , 然后 放 在 某 个 地 方 , 这 样 可 能 会 
有 一 些 隐私 的 信息 被 第 三 方 看 到 。 另 外 ， 用 户 也 可 以 在 浏览 器 上 直接 看 到 提交 的 数据 ， 一 些 系统 内 
部 消息 将 会 一 同 显 示 在 用 户 面前 。Post 的 所 有 操作 对 用 户 来 说 都 是 不 可 见 的 。 

(4) Get 传输 的 数据 量 小 ， 这 主要 是 因为 受 URL 长 度 限 制 ; 而 Post 可 以 传输 大 量 的 数据 ， 所 以 
在 上 传 文件 时 只 能 使 用 Post。 

(5) Get 限制 Form 表单 的 数据 集 的 值 必须 为 ASCII 字符 ;而 Post 支持 整个 ISO 10646 字符 集 。 

这 里 需要 注意 的 是 ，Get 是 Form 的 默认 方法 ， 但 通常 情况 下 希望 使 用 Post， 因 为 它 能 处 理 更 多 
的 数据 , 也 就 是 使 用 表单 插入 和 修改 大 块 文本 时 能 处 理 更 多 数据 。 如 果 使 用 Post, 则 所 有 发 送 给 PHP 
脚本 的 提交 数据 都 必须 使 用 $_ POST 语法 来 引用 ; 如果 使 用 Get， 则 需要 使 用 $_GET 语法 来 引用 。 


4.4.2 ”获取 表单 提交 数据 


PHP 作为 一 种 Web 程序 设计 的 脚本 语言 ， 它 能 使 用 户 很 容易 地 获取 HTML 表单 提交 的 数据 。 
4.4.1 小 节 介 绍 了 HTML 中 Form 表单 的 两 种 数据 提交 方式 : Get 和 Post， 接 下 来 列 出 两 个 案例 来 展 
示 一 下 它们 是 如 何 获取 表单 提交 的 数据 的 。 

为 了 和 用 户 进行 交互 ， 让 用 户 在 浏览 网 页 时 填写 并 且 提 交 一 些 表单 是 必需 的 。 例 如 在 一 个 虚拟 
社区 中 必须 要 有 用 户 登 录 的 表单 以 及 用 户 注册 的 表单 等 。 而 表单 的 数据 就 是 使 用 这 两 种 方式 之 一 传 
送 给 PHP 脚本 程序 进行 处 理 的 。 
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1. 使 用 Post 方 式 
下 面 使 用 Post 方式 提交 数据 ,该 页 面 postlogin.php 用 于 收集 用 户 登 录 的 信息 , 并 以 Post 方式 提 
交 给 showpostphp 页 面 进行 处 理 。 它 的 具体 代码 如 下 所 示 : 


showpost.php 页 面 用 于 获取 postlogin.php 页 面 以 Post 方式 提交 的 数据 ， 并 显示 在 该 页 面 上 。 它 
的 具体 代码 如 下 所 示 : 


Sname = $ POST['username']; 


Spass = $ POST['pass']; 


2 

你 输入 的 用 户 名 是 : <?php echo $name; ?> 

<br /> 3 

你 输入 的 密码 是 : <?php echo $pass; ?> 9 

</center> 

<body> 

</body> 

</html> 

在 上 述 代码 中 ， 使 用 $_POSTU] 来 获取 由 前 一 个 页 面 通过 Post 方式 提交 的 数据 ， 其 中 ,“[]” 中 第 
包含 要 获取 数据 的 name 属性 值 。 4 

把 上 述 文件 postlogin.php 和 showpost.php 存储 在 Apache\htdocs4 子 目录 下 ， 打 开 正 浏览 器 ， 3 
在 地 址 栏 中 输入 http://localhost/4/postlogin.php， 按 Enter 键 会 显示 如 图 4-1 所 示 的 窗口 。 浊 


在 该 窗口 中 输入 用 户 名 “ 宋 岩 岩 ” 密码 “syy711” 用 户 信息 ， 然 后 单 击 【 确 认 登 录 】 按 钮 ， 会 
显示 如 图 4-2 所 示 的 窗口 。 
避 Port 提 交 方 式 的 使 用 - Wicrosoft Internet Erplorer 
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欢迎 登录 文件 中 查看 W) 收藏 Wj 工具 0) 条 肌 o 


= 二 让 四 ) | 简 htty: /jlocalhos V4/shewpost phy 

用 户 省 ， 区 ] 

E32 [二 】 你 输入 的 用 户 各 是 ， 宁 类 央 
你 输入 的 密码 是 ，2yy711 

HES La 
EEC 
图 4-1 用 户 登 录 图 4-2 显示 用 户 登录 信息 
2. 使 用 Get 方式 


把 页 面 postloginphp 中 的 Form 表单 的 method 属性 设置 为 “get”、action 属性 设置 为 
“showget.php”， 保存 页 面 为 getlogin.php。 其 中 ，showget.php 页 面 完 成 与 showpostphp 页 面 同样 的 
功能 ， 它 的 具体 代码 如 下 所 示 : 


案例 4-4 

<html> 

<head> 

<meta http-equiv="Content-Type" content="text/html; charset=gb2312" /> 

<title> 获 取 Get 方式 提交 的 数据 ， 并 显示 </title> 

</head> 

<center> 

<?php 
Sname 
Spass = $ GET['pass']; 


$_GET['username']; 


2 


你 输入 的 用 户 名 是 : <?php echo $name; ?> 


93 
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<br /> 

你 输入 的 用 户 名 是 : <?php echo Spassy ?> 
</center> 

<body> 

</body> 

</html> 


这 里 要 注意 ， 在 上 述 代码 中 ， 使 用 $_GET[] 来 获取 由 前 一 个 页 面 通过 Get 方式 提交 的 数据 ， 其 
中 ,“[]” 中 包含 要 获取 数据 的 name 属性 值 。 

把 上 述 文件 getlogin.php 和 showget.php 存储 在 Apache\htdocs\4 子 目录 下 ， 打 开 正 浏览 器 ， 在 
地 址 栏 中 输入 http:Wlocalhost/4/getlogin php， 按 Enter 键 会 显示 类 似 于 如 图 4-1 所 示 的 。 然 后 在 窗口 
中 输入 用 户 名 “ 龙 建华 ” 密码 “ljh711” 用 户 信息 ， 然 后 单 击 【确认 登录 】 按 钮 ， 会 显示 如 图 4-3 
所 示 的 窗口 。 
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你 输入 的 用 户 名 是 ， 宛 建华 
你 辆 入 的 用 户 名 是 ，1jh711 


图 4-3 获取 Get 方式 提交 的 信息 


为 了 更 好 地 理解 Get 方式 的 数据 提交 ， 可 观察 一 下 图 4-3 中 的 地 址 栏 。 因 为 Get 方式 提交 的 数 
据 是 附加 在 页 面 URL 后 进行 传递 的 , 用 户 可 以 很 清楚 地 看 到 传递 的 信息 , 所 以 该 种 方式 传递 数据 存 
在 严重 的 安全 隐患 。 而 使 用 Post 方式 进行 提交 时 ，PHP 将 所 有 的 表单 字段 的 信息 放 到 一 个 叫做 
HITP POST VARS 的 数组 中 ， 然 后 由 数组 传递 给 脚本 程序 ， 所 以 在 提交 表单 时 不 会 看 到 浏览 器 的 
地 址 栏 中 出 现 图 4-3 中 的 URL。 

3. 获取 多 值 表单 的 数据 

前 面 介绍 了 单 值 表单 组 件 ， 它 的 值 很 容易 获 
取 。 那 么 如 果 一 个 表单 有 多 个 值 〈 多 值 表单 )， 该 
如 何 获取 表单 的 值 呢 ? 如 常见 的 复 选 框 和 多 选 
框 ， 它 们 有 效 地 提高 了 Web 的 数据 收集 能 力 ， 因 
为 这 些 多 值 组 件 多 许 用 户 为 一 个 指定 的 表单 项 同 
时 选择 多 个 值 。 例 如 ， 一 个 调查 用 户 经 常 使 用 哪 
些 编程 语言 的 表单 ， 可 以 通过 使 用 复 选 框 或 多 选 
框 来 表示 ， 该 表单 可 能 如 图 4-4 所 示 。 

图 4-4 中 的 部 分 HTML 代码 如 下 所 示 : 

案例 4-5 

<table width="257" align="center" bgcolor="#CCCCCC"> 

<tr> 


<td> 复 选 框 </td> 
<td> 
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4-4 ”使 用 两 种 方式 表示 相同 的 数据 


<input name="languages" type="checkbox" value="csharp">C#<br> 


在 上 述 代码 中 使 用 languages 来 引用 多 个 语言 选项 ， 这 就 要 求 表单 处 理 函 数 必须 能 够 识别 一 个 
表单 变量 中 可 能 有 多 个 值 。 在 PHP 中 为 了 识别 赋 给 一 个 表单 的 多 个 值 , 将 其 考虑 为 数组 而 增加 一 对 
中 括号 。 因 此 ， 表 单项 名 将 不 是 languages， 而 是 languages[]。 下 面 列 出 一 个 完整 案例 来 展示 如 何 获 
取 多 值 表 单 的 数据 。 该 案例 的 具体 代码 如 下 所 示 : 


监 流 “ 山 和 加 


把 上 述 代 码 存储 在 Apache\htdocsV 子 目录 下 的 mvphp 文件 中 ,打开 正 浏览 器 ， 在 地 址 栏 中 输 
入 http://localhost/4/mv.php， 按 Enter 键 会 显示 类 似 于 图 4-4 所 示 页 面 中 的 多 选 框 。 然 后 按 Shift 键 ， 
并 单 击 想 要 选择 的 项 ， 这 里 选择 “JAVA” 和 “PHP”。 单 击 【提交 】 按 钮 ， 会 显示 如 图 4-5 所 示 的 
窗口 。 
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图 4-5 获取 并 显示 多 值 表单 的 数据 


在 后 面 的 章节 中 ， 所 有 的 用 户 交互 界面 都 要 使 用 表单 来 实现 ， 而 表单 信息 的 处 理 都 要 利用 类 似 
于 以 上 案例 的 方法 来 实现 ， 所 以 读者 对 以 上 的 使 用 方法 要 熟练 掌握 。 


第 5 章 面向 对 象 的 PHP 
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面向 对 象 是 PHP 5.0 的 一 个 显著 特征 。 在 PHP 5.0 中 ，PHP 处 理 对 象 部 分 的 内 核 完全 重新 开发 
过 ,因此 在 提供 更 多 功能 的 同时 也 提高 了 性 能 ， 如 PHP 对 接口 的 处 理 方式 。 本 章 将 从 面向 对 象 的 特 
点 入 手 ， 对 PHP 的 面向 对 象 特性 进行 详细 的 介绍 。 
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了 解 PHP 中 的 面向 对 象 

了 解 PHP 面向 对 象 的 特点 

掌握 在 PHP 中 创建 类 和 对 象 

掌握 在 PHP 中 创建 和 使 用 字段 
掌握 在 PHP 中 使 用 属性 和 常量 
掌握 在 PHP 中 创建 和 使 用 方法 

掌握 PHP 的 构造 函数 和 析 构 函数 
掌握 PHP 静态 成 员 的 创建 和 使 用 
熟练 掌握 在 PHP 中 类 /对 象 函 数 的 使 用 


5.1 OOP 特性 


面向 对 象 程序 设计 “Object-Oriented Programming，OOP) 是 一 种 通过 使 用 对 象 来 解决 问题 的 技 
术 。 在 编程 术语 中 ， 对 象 是 一 些 实体 ， 它 们 的 特性 和 行为 是 解决 问题 所 必需 的 。 在 程序 的 开发 策略 
中 ， 把 完成 某 一 部 分 的 功能 代码 ， 封 装 成 一 个 类 ， 以 方便 其 他 的 程序 在 执行 相同 的 功能 时 调用 ， 达 
到 代码 重用 的 目的 。OOP 的 产生 ， 使 编程 的 注意 力 重 新 从 应 用 程序 的 逻辑 回 到 了 数据 上 来 ， 换 句 话 
说 ，OOP 将 焦点 从 编程 的 过 程 性 事件 转向 最 终 建 模 的 真实 实体 ， 面 向 对 象 的 编程 语言 更 加 接近 了 周 
围 的 现实 世界 。 在 学 习 PHP 面向 对 象 的 特性 之 前 ， 要 先 了 解 OOP 的 概念 。 本 节 介绍 OOP 的 3 个 基 
本 概念 : 封装 、 继 承 和 多 态 。 


5.1.1 封装 
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在 现实 世界 中 ， 可 以 把 身边 的 一 切 作为 一 个 对 象 来 看 待 ， 如 办 公 室 的 电脑 桌 、 键 盘 、 一 本 书 、 
马路 上 奔跑 的 汽车 、 路 上 行走 的 行人 。 这 些 存在 的 实体 ， 都 在 执行 自己 的 动作 ， 或 者 以 某 种 形态 存 
在 。 整 个 世界 是 由 无 数 个 独立 的 实体 组 成 的 。 我 们 不 需要 了 解 每 个 实体 的 创建 过 程 、 运 行 方式 ， 如 
汽车 是 一 个 实体 ， 汽 车 的 方向 盘 是 用 什么 材料 制 成 的 ， 它 的 制作 流程 是 什么 ， 只 需要 知道 可 以 通过 
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方向 盘 控 制 汽车 的 方向 ， 怎 样 使 用 它 就 可 以 了 。 面 向 对 象 的 程序 设计 的 思维 模式 就 是 调用 另外 一 个 
存在 的 实体 , 不 需要 知道 该 实体 的 实现 细节 , 只 需要 知道 使 用 该 实体 的 通用 接口 。 在 OOP 的 程序 中 ， 
存在 着 无 数 个 类 似 于 现实 世界 中 的 实体 ， 这 些 实体 称 为 类 。 在 每 一 个 实体 中 ， 都 会 存在 相应 的 行为 
和 特点 ， 其 他 的 实体 可 以 通过 相应 的 接口 来 操作 这 个 实体 。 

封装 是 类 的 基础 ， 即 把 类 的 相关 实现 细节 隐藏 起 来 。 调 用 这 些 类 时 ， 直 接 使 用 类 预 留 的 接口 就 
可 以 了 ， 如 电视 机 ， 我 们 不 知道 里 面 实现 的 细节 ， 怎 样 把 信号 转换 为 视频 和 音频 ， 但 是 只 要 知道 ， 
怎样 打开 电视 ， 怎 样 调节 音量 ， 怎 样 选择 电视 频道 就 可 以 了 。 封 装 把 用 户 使 用 的 接口 和 里 面 实现 的 
细节 分 开 了 。 

面向 对 象 的 程序 设计 通过 建立 良好 定义 的 接口 ， 使 隐藏 应 用 程序 内 部 工作 原理 的 概念 得 到 进 一 
步 提升 , 每 个 应 用 程序 的 组 件 都 可 以 访问 这 个 接口 。 牢记 OOP 思想 的 开发 人 员 不 会 陷入 大 量 的 细节 
之 中 ， 而 会 设计 出 独立 于 其 他 组 件 的 应 用 程序 组 件 ， 这 些 组 件 不 仅 允 许 重 用 ， 还 能 使 开发 人 员 组 合 
这 些 组 件 ， 而 不 是 将 组 件 紧 密 地 结合 或 耦合 在 一 起 。 通 过 这 些 良 好 定义 的 接口 进行 交互 的 组 件 称 为 
对 象 。 对 象 是 通过 类 的 实例 化 创建 而 成 的 。 类 用 于 定义 期 望 某 个 实体 所 具有 的 数据 和 行为 。 类 通过 
一 些 函数 提供 某 些 行 为 ， 这 些 函 数 称 为 方法 ， 方 法 主要 用 于 处 理 类 中 的 属性 ， 这 些 属性 称 为 字段 。 
类 是 一 个 概念 模型 ， 只 是 定义 了 一 些 函 数 和 字段 。 对 象 是 一 个 现实 存在 的 模型 。 


5.1.2 ”继承 


在 现实 世界 中 ， 一 个 公司 中 有 很 多 不 同 的 职位 ， 如 职员 、 主 管 和 出 纳 等 ， 这 些 职位 都 有 自己 相 
应 的 岗位 规则 ， 可 以 做 什么 ， 不 可 以 做 什么 等 。 这 些 概念 是 一 些 定义 ， 如 果 成 为 该 公司 的 员工 ， 就 
要 遵守 这 些 岗 位 的 规则 。 可 以 这 样 认 为 ,普通 职员 是 一 个 类 ,主管 是 一 个 类 ， 都 是 行使 不 同 的 职责 。 
主管 首先 必须 是 该 公司 的 职员 ， 然 后 才 是 一 个 管理 者 ， 也 就 是 说 主管 类 继承 了 职员 类 。 主 管 类 不 但 
有 具 有 职员 类 的 行为 和 特点 ， 还 要 具有 主管 类 本 身 的 一 些 行为 和 特点 。 在 程序 中 ， 如 果 一 个 类 继承 另 
外 一 个 类 ， 那 么 该 类 就 会 具有 父 类 的 方法 和 属性 ， 除 了 父 类 的 私有 成 员 。 实 际 上 ， 还 存在 另外 一 种 
形式 的 继承 ， 即 一 个 类 只 继承 了 一 些 抽象 的 概念 ， 这 些 概念 是 在 类 中 实现 的 。 

目前 PHP 不 支持 多 重 继承 ， 只 支持 单 重 继承 ， 所 以 不 能 从 两 个 或 两 个 以 上 类 中 派生 新 的 类 。 面 
向 对 象 的 开发 方法 建立 在 继承 概念 的 基础 上 。 这 种 策略 提高 了 代码 的 可 重用 性 ， 因 为 它 使 得 人 们 能 
够 在 多 个 应 用 程序 中 使 用 良好 设计 的 类 。 继 承 的 使 用 ， 也 提高 了 类 之 间 的 层次 性 。 


5.1.3 多 态 


多 态 性 (来 自 希腊 语 ， 表 示 “ 很 多 形态 ”) 是 使 一 个 对 象 被 看 成 另 一 个 对 象 的 技术 。 比 如 ， 有 
一 个 绵羊 牧场 , 里面 有 4 只 绵羊 (绵羊 属 ), 但 是 刚刚 买 了 两 只 山羊 (山羊 属 ) 和 一 只 德国 牧羊 犬 (大 
科 犬 属 )。 一 共有 多 少 动物 ?把 所 有 的 绵羊 、 山 羊 和 狗 加 起 来 ， 结 果 是 7 只 。 这 刚好 应 用 了 多 态 性 ， 
即 为 了 计算 ， 把 3 种 不 同 种 类 的 动物 当成 一 种 通用 类 型 (“动物”) 对待 。 如 果 把 绵羊 、 山 羊 和 犬 当 
成 哺乳 动物 看 待 ， 这 就 是 一 个 简单 的 信息 飞跃 。 生 物 学 家 每 天 都 以 这 种 方式 使 用 多 态 性 ， 而 程序 员 
则 以 从 其 他 科学 领域 “ 窃 用 ”(“ 重 用 ”) 好 主意 闻名 。 映 射 到 程序 中 ， 可 以 使 用 一 个 对 象 来 完成 不 同 
环境 下 的 功能 操作 。 

简单 地 讲 ， 多 态 是 指 OOP 能 够 根据 使 用 类 的 上 下 文 来 重新 定义 或 改变 类 的 性 质 或 行为 。 多 态 


是 对 象 的 一 种 能 力 ， 它 可 以 在 运行 时 根据 传递 的 对 象 参数 ， 决 定 调用 哪 一 个 对 象 的 方法 。 例 如 ， 如 
果 有 一 个 figure 的 类 ， 它 定义 了 一 个 draw 的 方法 ， 并 且 派 生 了 circle 和 rectangle 类 ， 在 派生 类 中 覆 
盖 了 draw 方法 ， 可 能 还 有 一 个 函数 ， 它 希望 使 用 一 个 参数 x， 并 且 可 以 调用 $x->draw0。 如 果 有 多 
态 性 ， 调 用 哪个 draw 方法 就 依赖 于 传递 给 这 个 函数 的 对 象 类 型 。 


5.2 关键 的 OOP 概念 


本 节 主 要 介绍 OOP 的 基础 性 概念 ， 如 类 的 创建 、 对 象 的 实例 化 等 ， 并 在 每 个 知识 点 后 ， 提 供 
了 相应 的 案例 。 


5.2.1 类 和 对 象 


面向 对 象 程序 设计 的 核心 思想 是 一 切 皆 是 对 象 ， 即 现实 世界 中 的 一 切实 体 都 可 以 看 作对 象 。 一 
个 人 就 是 一 个 对 象 ， 但 是 人 可 以 划分 不 同 的 类 别 ， 如 儿童 、 少 年 、 青 年 等 。 那 么 在 这 里 ， 儿 童 就 是 
一 个 类 , 儿童 的 具体 概念 就 是 年 龄 在 10 岁 以 下 的 人 ,同样 少年 也 是 一 个 类 别 。 类 是 具有 相同 行为 和 
特点 的 多 个 对 象 的 集合 、 如 图 形 类 、 哺 乳 动物 类 等 。 

同样 ， 在 PHP 程序 中 ， 也 可 以 创建 一 个 类 ， 这 时 类 只 是 具有 一 些 概念 的 模型 ， 即 类 中 定义 了 相 
应 的 方法 和 字段 ， 而 没有 实现 这 些 概念 。 在 PHP 中 创建 一 个 类 的 语法 格式 如 下 所 示 : 


class Name 

Hi 
Access Variable Declaration 
Access Function Declaration 


上 


在 上 述 代 码 中 ， 类 名 是 Name， 类 中 的 语句 包含 在 一 对 大 括号 中 ， 一 个 类 由 成 员 方法 和 成 员 字 
段 组 成 。 类 的 创建 示例 如 下 所 示 : 


<?php 
class Cart { 
public $items; // 购物 车 中 的 物品 
// 将 Snum 个 Sartnr 物品 加 入 购物 车 
function add item($artnr, S$num) { 
Sthis->items [Sartnr] += $num; 
} 
// 将 Snum 个 $artnr 物品 从 购物 车 中 取出 
function remove item($artnr, $num) { 
if ($this->items[$artnr] > $num) { 


$this->items[$artnr] -= $num; 
return true; 
} elseif (Sthis->items [Sartnr] == Snum) { 


unset ($this->items[S$artnr]); 
return true; 
} else { 
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return false; 


1 

2> 

上 述 代 码 定义 了 一 个 Cart 类 , 这 个 类 由 购物 车 中 的 商品 构成 的 数组 和 两 个 用 于 从 购物 车 中 添加 
和 删除 商品 的 函数 组 成 。 

类 实际 上 是 一 个 模板 , 它 定义 了 某 个 概念 或 真实 事物 的 性 质 和 行为 。 这 个 模板 提供 了 一 个 基础 ， 
可 以 在 此 基础 上 创建 实体 〈 即 依据 该 模板 建立 的 实体 ) 的 特定 实例 ， 这 些 特 定 的 实例 称 为 对 象 。 对 
象 是 类 的 概念 实现 ， 是 类 实例 化 后 的 结果 ， 如 一 只 老虎 就 是 哺乳 类 的 一 个 实例 化 对 象 ， 老 虎 具 有 哺 
乳 类 动物 的 行为 和 特点 。 

在 PHP 中 ， 可 以 使 用 new 关键 字 创 建 一 个 对 象 ， 其 语法 格式 如 下 所 示 : 


$0Obj=new Cart(); 


当 对 象 创建 完成 之 后 ， 就 可 以 直接 调用 类 中 的 方法 和 字段 。 一 个 类 可 以 创建 多 个 对 象 。 
5.2.2 ”字段 


在 PHP 中 ,字段 用 来 描述 类 的 某 个 方面 的 性 质 。 它 与 一 般 的 变量 非常 相似 ， 只 是 有 一 些 细微 的 
差别 。 
1. 声明 字段 
类 中 的 字段 实际 上 是 对 象 所 具有 的 属性 , 用 来 表示 实体 的 某 一 种 状态 。 在 PHP 中 创建 一 个 字段 
和 创建 一 个 变量 基本 相同 ， 只 不 过 创建 的 位 置 不 同 罢 了 。PHP 是 一 种 弱 类 型 的 语言 ， 字 段 可 以 不 需 
要 声明 ， 可 以 直接 创建 并 赋值 ， 但 不 推荐 这 样 使 用 。 也 可 以 在 类 中 先 声 明 ， 再 赋值 。 通 常 都 是 在 类 
的 开始 声明 字段 ， 并 为 字段 赋 初 值 。 声 明 字段 的 示例 如 下 所 示 : 
class Stri{ 
public $name="john"; 
private $age; 
$age=35; 
} 
在 上 述 代码 中 ， 创 建 了 两 个 字段 name 和 age。public 和 private 表示 这 两 个 字段 的 作用 域 范围 。 
第 一 种 方式 是 在 创建 变量 时 就 赋 初 值 ， 第 二 种 方式 是 变量 先 声 明 ， 后 赋 初 值 。 
2. 使 用 字段 
与 变量 不 同 的 是 ， 调 用 字段 不 是 利用 $ 符 号 ， 而 是 利用 -> 操作 符 。 字 段 只 属于 某 一 个 对 象 ， 故 在 
使 用 字段 时 ， 需 要 指明 该 字段 属于 哪个 对 象 。 使 用 字段 的 语法 格式 如 下 所 示 : 
Sobject->field 


字段 的 使 用 示例 如 下 所 示 : 
<?php 


class Hello{ 
public $str-" 欢 迎 使 用 PHP 中 的 类 处 理 问题 "; 
public $aa=" 请 调用 字段 "; 

} 

SHe=new Hello(); 

echo $He->str; 

echo "<br>"; 

echo $He->aa; 

2 


在 上 述 代码 中 创建 了 一 个 类 Hello， 并 在 类 中 创建 了 两 个 字段 ， 分 别 为 sr 和 aa。 在 下 面 创建 了 
该 类 的 对 象 He， 并 使 用 操作 符 -> 把 对 象 中 的 字段 分 别 输出 。 
在 定义 字段 的 类 本 身 引 用 该 字段 时 ， 还 是 需要 使 用 -> 操作 符 。 但 是 此 时 不 使 用 相应 的 对 象 名 或 
者 类 名 ， 而 是 使 用 this 关键 字 。this 关键 字 表示 所 使 用 的 字段 为 当前 类 所 具有 的 字段 。 其 使 用 示例 
如 下 所 示 : 
<?php 
Class Suf{ 
public $numl=5; 
public $num2=3; 
function sum(){ 
$total=$this->numl+$this->num2; 
echo $total; 


1 
} 
$s=new Su(); 
$s->sum(); 
?> 
在 上 述 代码 中 创建 了 一 个 类 Su， 在 该 类 中 ， 创 建 了 两 个 变量 numl 和 num2， 并 定义 了 一 个 方 
法 求 取 这 两 个 变量 的 和 ， 在 方法 中 调用 类 本 身 的 字段 时 ， 使 用 了 this 关键 字 。 
3. 字段 的 作用 域 
在 PHP 中 , 字段 共有 三 个 相应 的 作用 域 范围 , 用 来 表示 类 的 字段 可 以 影响 的 空间 , 其 值 分 别 为 : 
public、private、protected。 其 详细 信息 如 表 5-1 所 示 。 
表 5-1 访问 控制 修饰 符 
关键 字 说 了 明 
public 表示 修饰 的 字段 的 作用 域 范围 是 公共 作用 域 
private | 表示 修饰 的 字段 只 能 在 类 中 访问 ， 类 以 外 的 其 他 位 置 无 法 访问 该 字段 


protected ”| 表示 修饰 的 字段 是 受 保护 的 。 受 保护 的 字段 只 能 在 类 中 调用 ， 不 允许 在 类 的 外 部 调用 这 些 字段 


。 public 该 作用 域 的 字段 可 以 在 类 外 部 直接 通过 对 象 名 访问 ， 并 进行 修改 。 其 使 用 示例 如 下 
所 示 : 
<?php 


class Ja{ 
public $a; 


} 
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在 上 述 代码 中 创建 了 类 Ja， 并 在 类 中 创建 了 一 个 具有 公共 作用 域 的 变量 a， 没 有 赋 初 值 。 在 类 
的 外 部 ， 创 建 类 Ja 的 对 象 J， 并 给 字段 a 赋值 为 5， 然 后 把 该 值 赋 给 一 个 变量 str 并 输出 该 值 。 

在 类 中 声明 某 个 字段 是 公共 作用 域 ， 方 便 了 以 后 对 该 字段 的 使 用 和 修改 。 但 是 在 OOP 中 并 不 
鼓励 使 用 公共 字段 , 因为 这 样 做 会 把 程序 的 一 些 细节 暴露 在 类 的 外 部 , 并 且 直 接 访 问 类 的 某 些 数据 ， 
会 忽略 对 某 些 数 据 的 数据 验证 。 例 如 无 法 阻止 用 户 修改 字段 的 值 。 对 于 这 样 的 情况 ， 通 常 采 用 两 种 
方法 解决 ， 一 种 方法 是 将 数据 封装 在 对 象 中 ， 只 通过 一 些 称 为 公共 方法 的 接口 来 访问 ， 这 种 封装 的 
数据 具有 私有 作用 域 。 第 二 种 方法 是 使 用 属性 。 

。 private private 又 称 私 有 的 访问 控制 修饰 符 ， 是 限制 最 为 严密 的 控制 关键 字 。 其 使 用 示例 如 

下 所 示 : 


在 上 述 代码 中 ,创建 了 两 个 私有 字段 name 和 telephone， 该 字段 不 能 通过 该 类 的 实例 化 对 象 来 
访问 ， 也 不 能 通过 该 类 的 子 类 来 访问 。 如 果 需 要 在 类 的 外 部 使 用 这 些 字段 的 值 ， 可 以 通过 保护 作用 
域 来 完成 。 私 有 字段 的 访问 必须 通过 公共 接口 来 访问 ， 这 符合 前 面 介绍 的 封装 概念 。 私 有 字段 的 使 
用 示例 如 下 所 示 : 


在 上 述 示 例 中 创建 了 类 staff， 并 在 类 中 创建 了 一 个 私有 字段 name， 创 建 了 一 个 公共 方法 
setName0， 用 来 设置 私有 字段 name 的 值 。 在 下 面 创建 该 类 的 对 象 sta， 并 调用 该 类 的 公共 接口 设置 
私有 字段 的 值 。 

。 protected 表示 修饰 的 字段 是 受 保护 的 。 受 保护 的 字段 只 能 在 类 中 调用 ， 不 允许 在 类 的 外 部 

调用 这 些 字 段 。 其 使 用 示例 如 下 所 示 : 


和 私有 字段 的 区 别 在 于 ， 在 继承 的 子 类 中 ， 可 以 访问 这 些 受 保护 的 字段 ， 这 是 私有 字段 不 具备 
的 。 如 果 在 子 类 中 ， 试 图 访问 父 类 中 的 私有 字段 ， 将 会 导致 致命 的 错误 。 因 此 ， 如 果 和 希望 扩展 〈 继 
承 ) 该 类 ， 就 应 当 使 用 保护 字段 而 不 是 私有 字段 。 


5.2.3 ”属性 


OOP 提供 了 强大 的 功能 ， 属 性 就 是 一 个 例子 ， 属 性 是 受 保护 的 字段 。 通 过 强制 在 方法 中 访问 和 
操作 字段 ， 一 方面 保护 字段 ， 另 一 方面 允许 像 访 问 字段 一 样 访 问 数据 。 这 些 方法 称 为 访问 方法 和 修 
改 方法 ， 或 非 正 式 地 称 为 获取 方法 (getxxx0)〉 和 设置 方法 (setxxx0)。 它 们 将 分 别 在 访问 或 操作 字 
段 时 自动 触发 。 

在 PHP 5.0 中 ,没有 提供 和 其 他 OOP 语言 一 样 处 理 并 使 用 属性 的 功能 。 如 果 要 实现 该 功能 ,不 
能 够 直接 实现 ， 而 需要 使 用 公共 方法 模拟 这 种 功能 。 需 要 声明 两 个 函数 getName0 和 setName0， 为 
属性 name 分 别 创建 获取 方法 和 设置 方法 ， 并 可 以 在 函数 中 设置 相应 的 语句 实现 功能 。 

在 PHP 5.0 中 , 通过 重 载 _set0 方 法 和 _get0 方 法 也 能 实现 对 属性 的 处 理 。 在 PHP 类 的 执行 过 
程 中 ， 如果 在 下 面 的 代码 中 , 试图 引用 一 个 类 定义 中 不 存在 的 成 员 变 量 时 , 就 会 自动 触发 这 些 方法 ， 
并 执行 方法 中 的 语句 。 属 性 有 很 多 用 途 ， 如 产生 报错 消息 ， 甚 至 通过 动态 创建 新 的 变量 来 扩展 类 。 

这 里 与 PHP 4.0 不 同 ， 在 原来 的 版 本 中 ， 直 接 使 用 一 个 没有 在 类 中 声明 的 例子 是 可 以 执行 的 ， 
然而 在 实际 使 用 中 并 不 希望 给 对 象 没 有 定义 的 属性 进行 操作 ， 和 希望 编程 人 员 能 够 严格 按照 所 设计 的 
结构 进行 , 这 时 PHP 4.0 就 无 能 为 力 了 。 但 是 该 问题 在 PHP 5.0 中 有 了 较 好 的 解决 , 即 __set 和 __get。 
下 面 利用 PHP 5.0 的 一 些 特性 改写 一 下 上 面 的 代码 。 

1. __set() 

该 方法 主要 负责 隐藏 字段 的 赋值 实现 ， 并 在 为 类 字段 赋值 之 前 验证 类 数据 。 它 接受 一 个 属性 名 
和 相应 的 值 作为 输入 ， 如 果 方 法 成 功 执行 就 返回 True， 否则 返回 False。 可 以 称 为 修改 方法 或 者 设置 
方法 。 该 方法 的 语法 格式 如 下 所 示 : 


boolean set([String property name], [mixed value to assign]) 


这 里 需要 注意 的 是 ，set0 函 数 前 面 的 下 划 线 是 两 个 而 不 是 一 个 ， 格 式 如 果 不 正确 ， 该 方法 是 不 
会 执行 的 。 下 面 创 建 一 个 案例 ， 演 示 一 下 该 方法 的 使 用 。 该 案例 的 代码 如 下 所 示 : 


案例 5-1 
<?php 
class staff{ 
var $name; 
function __set($propName,$propValue){ 
echo ' 该 变量 不 存在 '; 
echo "<br>"; 


1 


} 
Sem=new staff(); 
$em->name="'john'; 
$em->title='hello';// 该 属性 不 存在 
echo $em->name; 
?> 
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将 上 述 代码 保存 ， 文 件 名 为 set.php， 并 保存 到 指定 的 位 置 。 打 开 下 浏览 器 ， 在 地 址 栏 中 输入 
http://localhost:8080/myweb/set.php， 单 击 【 转 到 】 按 钮 ， 会 显示 如 图 5-1 所 示 的 窗口 。 
在 本 案例 中 ， 创 建 了 一 个 类 staff， 在 类 中 创建 了 一 个 属 
性 name， 并 重 写 了 方法 __set0， 用 来 检验 要 进行 赋值 的 变 
量 是 否 存在 ， 如 果 该 变量 不 存在 ， 就 会 执行 set( 方 法 中 的 
语句 。 然 后 创建 了 该 类 的 一 个 对 象 ， 并 给 name 以 及 不 存在 
的 属性 tile 赋 值 , 当 程序 执行 到 此 处 时 ,就 会 触发 执行 setO 


TITOTETTIETTITI 
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class foo { 


方法 。 图 5-1 __set0 方 法 的 使 用 
2. __get() 

| 该 方法 主要 负责 封装 获得 类 变量 所 需 的 代码 。 它 接受 一 个 属性 名 作为 输入 参数 ， 即 获取 属性 的 
值 。 它 在 成 功 执行 时 返回 True， 否 则 返回 False。 可 以 称 作 修 改 方法 和 获取 方法 。 该 方法 的 语法 格式 
过 如 下 所 示 : 
名 
四 boolean get([string property name]) 
攻 该 方法 的 使 用 示例 如 下 所 示 : 
习 <?php 
手 
册 


function __get($name) 
{ 
echo "现在 需要 获取 属性 $name"; 
} 
} 
$x = new foo(); 
$x->bar = 37 
print ($x->winky winky); 
2 


在 上 述 代码 中 创建 了 一 个 类 foo， 并 在 类 中 实现 了 __get0 方 法 ， 如 果 该 方法 被 触发 ， 就 执行 语 
句 。 在 类 的 下 面 当 试图 输出 一 个 没有 经 过 定义 的 属性 的 值 时 ， 就 会 触发 。_get0 方 法 。 

3. 创建 自 定义 获取 方法 和 设置 方法 

使 用 _set0 和 __get() 方 法 可 以 对 属性 进行 支持 , 但 对 于 管理 复杂 的 面向 对 象 应 用 程序 中 的 属性 
而 言 ， 它 们 的 功能 还 是 不 完善 。 可 以 采用 另外 一 种 方式 来 设置 和 获取 字段 的 值 。 当 在 一 个 类 中 创建 
一 个 变量 时 ， 可 以 为 该 变量 创建 两 个 方法 实现 变量 值 的 设置 和 获取 。 其 使 用 示例 如 下 所 示 : 


<?php 
class Exa{ 

private $name; 

public function getName(){ 
return $this->name; 

1 
public function setName ($name){ 
$this->name=$name; 

} 
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} 

$E=new Exa(); 

$E->setName (3); 

echo $E->getName (); 
> 


在 上 述 代 码 中 创建 了 一 个 类 Exa,， 并 在 类 中 创建 了 一 个 变量 name， 为 该 变量 创建 了 设置 和 获取 
方法 ， 即 setName0 和 getName0 方 法 。 然 后 在 类 外 调用 相应 的 设置 和 获取 方法 。 这 种 方式 不 如 使 用 

属性 那么 方便 ， 但 确实 通过 使 用 标准 的 命名 约定 封装 了 管理 和 存 取 任务 。 同 时 ， 还 可 以 向 设置 方法 

中 加 入 一 些 校 验 的 代码 。 


5.2.4 ”常量 


常量 顾名思义 就 是 在 程序 执行 过 程 中 ， 不 会 改变 的 数值 。 对 于 类 中 的 任何 一 个 实例 化 对 象 ， 在 
其 运行 过 程 中 ， 常 量 就 会 一 直 存在 ， 常 量 值 在 这 些 对 象 的 整个 声明 周期 中 都 会 保持 不 变 。 

可 以 在 类 中 创建 常量 ， 但 不 能 够 使 用 $ 符 号 去 声明 或 使 用 它 ， 需 要 使 用 const 关键 字 声 明 。 常 量 
不 同 于 普通 变量 ， 常 量 值 不 能 通过 对 象 的 实例 来 访问 ， 而 应 使 用 $object::constant 表达 式 访问 。 常 量 
值 必须 是 一 个 常量 表达 式 ， 而 不 是 一 个 变量 、 一 个 类 的 成 员 、 一 个 数学 表达 式 或 函数 调用 的 结果 。 

在 PHP 类 中 ， 创 建 一 个 常量 并 调用 常量 的 格式 如 下 : 


class MyClass 
{ const constant = 'constant Value'7 
function showConstant() { echo self::constant."\n"; } 
} 
echo MyClass::constant."\n"; 
$class = new MyClass(); 
$class->showConstant ();// echo $class::constant; is not allowed 
2 


上 述 代 码 中 ， 在 类 MyClass 中 创建 了 一 个 常量 和 方法 ， 当 在 方法 中 调用 这 个 常量 时 ， 不 能 使 用 
this 关键 字 ， 而 应 使 用 self 关键 字 。 类 常量 的 使 用 示例 如 下 所 示 : 


<?php 
class math fun{ 
const a='hello'; 
function showA(){ 
echo self::a; 
上 
} 
echo math fun::a; 
Sm=new math fun(); 
$m->showA(); 
> 


在 上 述 代码 中 创建 了 一 个 类 math fun, 并 定义 了 它 的 一 个 常量 a, 赋值 为 hello, 在 类 方法 showA 
中 调用 了 该 常量 a。 同 时 在 类 外 使 用 了 两 种 方式 调用 常量 。 一 种 是 通过 math fun::a 语句 实现 ， 另 外 
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一 种 是 通过 间接 方法 实现 。 
5.2.5 方法 


一 个 类 的 方法 (method) 实际 上 和 PHP 脚本 程序 中 的 函数 比较 相似 ， 只 不 过 方法 是 用 来 定义 类 
的 行为 。 类 中 的 方法 可 以 完成 指定 的 功能 ， 并 具有 返回 值 ， 同 样 也 可 以 接受 一 个 输入 的 参数 ， 并 对 
该 数值 进行 校 验 ， 返 回 结果 。 在 PHP 页 面 中 ， 调 用 一 个 函数 可 以 直接 通过 函数 名 来 实现 ， 调 用 类 的 
方法 和 调用 函数 一 样 ， 只 不 过 前 面 需要 加 上 对 象 名 。 

1. 声明 方法 

可 以 在 类 中 创建 一 个 方法 ， 其 语法 格式 和 函数 的 创建 相同 ， 只 不 过 类 中 的 方法 都 必须 使 用 关键 
字 public、protected 或 private 进行 定义 。 如 果 没 有 设置 这 些 关 键 字 ， 则 该 方法 会 被 设置 成 默认 的 
public。 声 明 方 法 的 语法 格式 如 下 所 示 : 


scope function methodName (){ 
方法 体 
| 


在 上 述 代码 中 ，scope 表示 方法 的 作用 域 范围 ，methodName 表示 方法 的 名 称 ， 大 括号 中 表示 该 
方法 的 执行 语句 。 该 方法 的 使 用 示例 如 下 所 示 : 
class Me{ 
private $value2=8; 
public function sum(Svaluel ){ 
return $valuel+$this->value2; 
} 
} 


在 上 述 代码 中 ,创建 了 一 个 类 Me， 在 类 中 定义 了 一 个 字段 value2， 声 明了 一 个 方法 sum0 求 取 
两 个 数 的 和 。 在 方法 sum0 中 使 用 this 关键 字 直 接 调 用 了 类 的 字段 value2， 通 过 将 参数 valuel 的 值 
和 value2 的 值 相 加 ， 返 回 结果 。 从 这 里 可 以 看 出 ， 类 中 的 方法 可 以 接受 外 部 的 参数 ， 并 可 以 调用 类 
本 身 的 字段 。 


0@ f 对 于 公共 方法 ， 可 以 不 显 式 声明 作用 域 ， 而 只 是 像 声明 函数 (没有 任何 作用 域 ) 那样 声明 方法 。 
示 

2. 调用 方法 

调用 方法 的 形式 非常 简单 ， 在 方法 名 前 面 加 上 对 象 名 就 可 以 了 。 对 于 上 面 的 示例 Me 中 的 方法 
sum()， 可 以 采用 下 面 的 形式 进行 调用 。 

$M=new Me(); 

echo $M->sum(4); 

首先 创建 该 类 的 对 象 M, 然后 通过 对 象 名 调用 方法 sam0， 并 向 方法 中 传递 一 个 参数 4， 调 用 操 
作 符 为 ->。 


3. 方法 作用 域 

PHP 中 方法 的 作用 域 关键 字 ， 常 见 的 有 public、private、protected， 除 了 这 些 作用 域 关键 字 外 ， 
修饰 方法 还 有 其 他 的 关键 字 ， 如 abstract、final、static 分 别 表示 方法 的 不 同 含义 。 方 法 的 常用 关键 
字 如 表 5-2 所 示 。 


表 5-2 方法 关键 字 二 


public 表示 该 方法 是 一 个 公共 方法 ， 在 PHP 页 面 的 任何 一 个 位 置 都 可 以 通过 对 象 名 来 访问 该 方法 

表示 该 方法 是 私有 的 ， 只 能 在 类 的 内 部 使 用 ， 不 能 通过 类 的 实例 化 对 象 调用 ， 也 不 能 通过 类 的 子 
类 调用 。 如 果 某 些 方法 是 作为 另外 一 些 方 法 的 辅助 ， 可 以 声明 该 方法 为 私有 的 
表示 方法 是 一 个 受 保护 的 方法 ， 只 能 在 类 本 身 和 类 的 子 类 中 使 用 。 这 类 方法 可 以 帮助 类 或 子 类 完 
成 内 部 计算 


Private 


protected 


第 

abstract 表示 方法 是 一 个 抽象 方法 。 抽 象 方法 表示 只 具有 方法 名 ， 而 没有 方法 体 的 方法 5 

表示 最 终 的 、 不 可 改变 的 含义 he 

面 

(1) public 向 

当 创 建 类 中 的 方法 时 ， 如 果 要 设置 该 方法 为 公共 方法 ， 就 需要 在 方法 的 前 面 加 上 public 关键 字 2 

或 者 省 略 不 写 。 下 面 创建 一 个 案例 演示 公共 方法 的 声明 和 调用 。 该 案例 的 代码 如 下 所 示 : 的 

案例 5-2 下 
<?php 


class Visitors{ 
public function greetVisitor(){ 
echo "Hello <br>"; 
3 
function sayGoodbye(){ 
echo "Goodbye<br>"; 
} 

} 
Visitors::greetVisitor(); 
$v=new Visitors(); 
$v->sayGoodbye () 7 

> 


将 上 述 代码 保存 ， 文 件 名 为 methodphp， 文 件 保存 在 C:\Web\apache\htdocs\class 目录 下 。 打 开 
下 浏览 器 ， 在 地 址 栏 中 输入 http://localhost:8080/class/ [ET 半 于 于 
method.php， 单 击 【 转 到 】 按 钮 ， 会 显示 如 图 5-2 所 示 | 3 本 和 2 于 志和 
的 窗口 。 

在 本 案例 中 ， 创 建 了 一 个 类 Visitors， 并 在 类 中 创 
建 了 两 个 方法 , 分 别 输出 不 同 的 字符 串 ， 这 两 个 方法 的 
作用 域 都 是 public。 下 面 在 调用 这 两 个 方法 时 采用 了 两 
种 不 同 的 形式 ， 一 种 是 通过 类 名 直接 调用 ， 如 语句 Visitors::greetVisitor0:， 其 调用 操作 符 为 ::。 另 外 
一 种 形式 是 通过 对 象 名 调用 ， 如 $v->sayGoodbye(;。 

(2) private 

如 果 要 创建 一 个 私有 方法 ， 只 需要 在 方法 的 前 面 加 上 private。 其 使 用 示例 如 下 所 示 : 


5-2 方法 调用 


串 汕 局 性 悄 滑 19SAW+dHd 


在 上 述 代 码 中 ， 创 建 了 一 个 类 pr， 并 创建 了 两 个 方法 va0 和 QO。 方法 va0 是 一 个 私有 的 方法 ， 
只 能 由 类 本 身 调用 。va0 方 法 充当 一 个 校 验 功 能 ， 对 参数 进行 判断 。 在 方法 Q0 中 ， 调 用 了 va0 方 法 
进行 判断 。 这 里 需要 注意 的 是 ， 在 一 个 方法 中 调用 另外 一 个 方法 ， 需 要 使 用 关键 字 this。 

(3) protected 

该 关键 字 表 示 方 法 是 一 个 受 保护 的 方法 。 例 如 下 面 的 示例 ， 在 获取 某 个 员工 信息 之 前 ， 可 能 需 
要 对 员工 信息 进行 校 验 , 它 将 作为 参数 传递 到 类 的 实例 化 方法 ( 即 构造 函数 ) 中 。 然 后 使 用 verify_ein0 
方法 验证 语法 是 否 正 确 。 因 为 这 个 方法 只 用 于 类 中 的 其 他 方法 ， 对 于 sta 企 派生 的 子 类 也 可 能 有 用 ， 
所 以 该 方法 应 声明 为 protected。 示 例 代 码 如 下 : 


试图 从 类 外 部 调用 protected 方法 ， 将 会 出 现 致命 性 错误 ， 该 方法 具有 保护 作用 。 

(4) abstract 

该 关键 字 可 以 用 在 类 和 方法 的 前 面 。 通 常 一 个 方法 被 声明 为 抽象 方法 时 ， 类 的 本 身 并 没有 实现 
该 方法 ， 一 般 是 由 类 的 子 类 来 实现 该 抽象 方法 。 如 果 在 一 个 类 中 ， 包 含 了 抽象 方法 ， 类 本 身 也 就 成 
为 了 一 个 抽象 类 ， 该 类 的 前 面 需要 带 有 关键 字 abstract。 抽 象 类 或 抽象 方法 定义 了 某 些 方面 的 共性 ， 
即 一 般 性 ， 主 要 实现 了 某 些 功能 的 命名 约定 。 抽 象 方法 的 语法 格式 如 下 所 示 : 


abstract function methodName ()7 
抽象 方法 的 应 用 示例 如 下 所 示 : 


abstract class Example{ 
abstract function Rbstrl()7 
abstract function Abstr2(); 
abstract function Abstr3(); 
| 


当 在 PHP 页 面 中 ， 创 建 了 一 个 抽象 类 时 ， 就 需要 在 Example 类 的 子 类 中 实现 这 些 抽象 方法 。 

(5) final 

该 关键 字 可 以 用 在 类 、 方 法 的 前 面 ， 这 些 成 员 统称 为 final 成 员 。 当 在 一 个 方法 的 前 面 加 上 final 
关键 字 时 ， 表 示 该 方法 不 可 以 被 重 写 ， 即 在 该 类 的 子 类 中 ， 只 允许 调用 ， 不 允许 重新 设置 该 方法 的 
功能 。final 方法 的 语法 格式 如 下 所 示 : 


class Example{ 
final function getValue(){} 


如 果 在 子 类 中 ， 试 图 重 写 getValue0 方 法 ， 会 导致 致命 性 错误 。 


5.3 ”构造 函数 和 析 构 函数 


当 类 创建 完成 后 ， 需 要 把 该 类 实例 化 才能 调用 该 类 的 成 员 方 法 和 成 员 变 量 。 一 个 类 的 实例 化 过 
程 ， 实 际 上 就 是 给 类 中 的 每 个 字段 分 配 相 应 的 内 存 空 间 ， 并 分 别 赋予 初 值 的 过 程 。 如 果 一 个 类 的 字 
段 很 多 ， 可 以 通过 构造 函数 (constructor) 来 实现 字段 的 初始 化 。 类 的 对 象 在 执行 过 程 中 ， 可 能 需要 
加 载 多 个 资源 ， 如 文件 对 象 、 数 据 库 对 象 等 、 程 序 执行 完毕 后 ， 如 果 不 释放 这 些 资源 就 会 永远 占有 ， 
时 间 过 长 ， 其 他 的 程序 不 能 使 用 这 些 占 有 资源 。 这 时 ， 需 要 通过 析 构 函数 〈destructor) 来 清除 字段 
或 者 对 象 占有 的 资源 。 本 节 将 详细 介绍 构造 函数 和 析 构 函数 的 创建 和 使 用 。 


5.3.1 构造 函数 


在 创建 类 的 实例 化 对 象 的 过 程 中 ， 往 往 希望 会 初始 化 字段 ， 或 执行 一 个 特殊 操作 ， 或 者 触发 一 
些 执行 函数 ， 这 些 操作 在 OOP 语言 中 一 般 是 自动 执行 的 。 这 样 的 机 制 在 OOP 中 称 为 构造 函数 ， 构 
造 函 数 主要 是 定义 对 象 实例 化 时 自动 执行 的 代码 。 

在 PHP 4.0 中 ， 当 函数 与 对 象 同名 时 ， 这 个 函数 将 成 为 该 对 象 的 构造 函数 。 在 PHP 5.0 中 ， 构 
造 函 数 被 统一 命名 为 ”_constmct。 如 果 在 一 个 类 中 声明 一 个 函数 ， 命 名 为 ”_constrct， 这 个 函数 将 
被 当 作 一 个 构造 函数 ， 并 在 建立 一 个 对 象 实例 时 被 执行 。 注 意 ， _ 是 两 个 下 划 线 。 就 像 其 他 任何 函 
数 一 样 ， 构 造 函数 可 能 有 参数 或 者 默认 值 。 可 以 定义 一 个 类 来 建立 一 个 对 象 ， 并 将 其 属性 全 部 放 在 
一 个 语句 (statement) 中 。 
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构造 函数 可 以 完成 下 面 的 一 些 操作 , 接受 参数 并 把 值 赋 给 特定 的 对 象 字段 , 调用 其 他 类 的 方法 ， 
调用 其 他 类 的 构造 函数 ， 包 括 它 自己 的 父 类 。 


构造 函数 的 语法 格式 如 下 所 示 : 
function construct ([argl,arg2...argn],){ 
方法 体 


} 


构造 函数 中 可 以 带 有 参数 ， 也 可 以 不 带 有 。 下 面 创建 一 个 案例 ， 演 示 构 造 函 数 的 创建 。 该 案例 
的 代码 如 下 所 示 : 


案例 5-3 
<?php 
class foo { 
private $x; 
private $y; 
function construct ($x) { 
S$this->x = $x; 
Sthis->getY() 7 
function display() { 
print($this->x); 
} 
function getY() { 
$y="bye bye"; 
echo $y; 
} 
站 
$01 = new foo("john") 7 
$01->display(); 
?> 


Ey 
ua 
三 
十 

< 
< 

四 
名 
| 
完 
全 
至 
习 
手 
册 


将 上 述 代码 保存 ， 文 件 名 为 constrphp， 并 保存 到 C:\Web\apache\htdocs\class 目录 下 ， 在 本 章 以 
后 对 于 文件 的 保存 位 置 不 再 特别 声明 ， 都 是 在 该 目录 下 
保存 。 打 开 正 浏览 器 ， 在 地 址 栏 中 输入 http://localhost: 
8080/class/constr.php， 单 击 【 转 到 】 按 钮 ， 会 显示 如 图 


http://localhost:8080/c 


5-3 所 示 的 窗口 。 
在 上 述 代码 中 创建 了 两 个 变量 x 和 y，x 在 构造 函 
数 中 直接 初始 化 ，y 在 方法 getY0 中 初始 化 并 输出 。 在 
构造 函数 中 , 调用 getY0 方 法 .在 下 面 创建 foo 类 的 对 象 ol 时 , 需要 给 构造 函数 传递 一 个 参数 “john 
并 利用 对 象 ol 调用 方法 display0。 

在 本 案例 中 ， 构 造 函数 调用 了 一 个 方法 getY0 来 实现 对 变量 y 的 操作 。 实 际 上 在 构造 函数 中 不 
但 可 以 调用 类 本 身 的 方法 ， 也 可 以 调用 其 他 类 的 构造 函数 。 调 用 其 他 类 的 构造 函数 的 语法 格式 如 下 
所 示 : 


图 5-3 构造 函数 


classname:: _construct() 
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classname 表示 要 调用 的 类 的 名 称 ， _construct0 表 示 类 的 构造 函数 ,“: : ”表示 调用 构造 函数 的 
操作 符 。 其 示例 如 下 所 示 : 


<?php 
class fool { 
function __construct($x) { 和 
echo $x; 
} 
ls 
class foo2 { 
function construct ($y) { 
fool:: construct("world"); 
echo $y; 
} 
} 
$fl=new fool ("hello"); 
$f2=new foo2("china"); 
?> 


在 上 述 代码 中 创建 了 两 个 类 ， 分 别 为 fool 和 foo2。 在 类 fool 的 构造 函数 中 输出 参数 x 的 值 
在 类 foo2 的 构造 函数 中 输出 参数 y 的 值 ， 并 调用 类 fool 的 构造 函数 。 需 要 注意 的 是 构造 函数 的 前 
面 要 加 上 两 个 下 划 线 。 

同样 ， 在 子 类 中 也 可 以 调用 父 类 的 构造 函数 ， 对 于 该 知识 点 的 介绍 会 在 后 面 的 章节 中 提 到 。 


5.3.2” 析 构 函 数 


对 于 面向 对 象 的 编程 来 说 ， 定 义 析 构 方法 是 一 个 非常 有 用 的 功能 。 析 构 方法 可 以 用 来 记录 调试 
信息 、 关 闭 数据 库 连接 等 一 些 清除 收尾 的 工作 。PHP 4.0 中 没有 析 构 方法 ， 尽 管 PHP 4.0 已 经 支持 可 
以 注册 一 个 函数 以 便 请 求 结束 时 调用 。PHP 5.0 引进 的 析 构 方法 的 概念 和 其 他 面向 对 象 的 语言 〈 比 
如 Java) 是 一 致 的 。 当 指向 这 个 对 象 的 最 后 一 个 引用 被 销毁 时 ， 析 构 方 法 被 调用 ， 调 用 完成 后 释放 
内 存 。 注 意 ， 析 构 方法 不 接受 任何 参数 。 

析 构 函数 __destruct0, 相反 于 构造 函数 。 PHP 调用 它 来 将 一 个 对 象 从 内 存 中 销毁 。 默认 情况 下 ， 
PHP 仅仅 释放 对 象 属性 所 占用 的 内 存 并 销毁 与 对 象 相关 的 资源 。 析 构 函 数 允 许 在 使 用 一 个 对 象 之 后 
执行 任意 代码 来 清除 内 存 。 

当 PHP 决定 的 脚本 不 再 与 对 象 相关 时 ， 析 构 函数 将 被 调用 。 在 一 个 函数 的 命名 空间 内 ， 这 会 发 
生 在 出 现 函数 retum0 时 。 对 于 全 局 变量 ， 这 发 生 在 脚本 结束 时 。 如 果 想 明确 地 销毁 一 个 对 象 ， 可 以 
给 指向 该 对 象 的 变量 分 配 任何 其 他 值 。 通 常 将 变量 赋值 为 NULL 或 者 调用 unset0 方 法 。 

析 构 函数 的 使 用 示例 如 下 所 示 : 

<?php 

class foo { 

Var $x; 

function construct ($x) { 
Sthis->x = $x; 


} 
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function destruct() { 
print ("bye bye"); 
上 
上 
S$ol = new foo(4)7 
2 


在 上 述 代码 中 创建 了 一 个 类 foo， 该 类 重 写 了 构造 函数 和 析 构 函数 。 脚 本 执行 结束 时 ，PHP 会 
撤销 内 存 中 的 所 有 对 象 。 因 此 ， 如 果实 例 化 的 类 和 实例 化 创建 的 信息 留 在 内 存 中 ， 就 不 需要 显 式 声 
明 析 构 函数 。 但 是 如 果实 例 化 时 创建 了 不 容易 丢失 的 数据 ， 就 应 当 在 对 象 撤销 时 撤销 这 些 数据 ， 为 
此 需要 创建 一 个 定制 的 析 构 函数 ， 即 重 写 析 构 函数 __destruct0。 


和》 析 构 通 数 在 肢 本 关闭 时 调用 ， 此 时 所 有 的 头 信息 已 经 发 出 ,试图 在 析 构 函数 中 抛 出 一 个 异常 会 导致 致 
提示 | 命 错误 。 


5.4 新 增 OOP 特性 


PHP 5.0 版 本 相对 于 以 前 的 版 本 ， 在 面向 对 象 方面 的 改动 较 大 ， 并 且 在 面向 对 象 的 设计 上 ， 又 
增加 了 一 些 新 的 特性 。 这 些 特性 的 产生 使 PHP 更 加 能 够 以 面向 对 象 程序 设计 的 思想 ， 设 计 PHP 程 
序 。 本 节 将 对 PHP 新 增 的 OOP 特性 进行 详细 的 介绍 。 


5.4.1 类 型 提示 


类 型 提示 是 PHP 5.0 中 的 一 种 新 特征 , 它 能 够 把 函数 参数 强制 转换 成 特定 类 型 的 对 象 . 在 PHP 5.0 
之 前 ， 唯 一 确保 函数 参数 是 一 个 特定 对 象 类 型 的 方法 是 使 用 PHP 提供 的 类 型 检查 函数 〈 即 is_a0 )。 
现在 ， 可 以 简单 地 强制 转换 对 象 类 型 ， 通 过 在 函数 参数 的 前 面 加 上 类 型 名 。 类 型 提示 的 使 用 示例 如 
下 所 示 : 


<?php 
class MyClass 
到 
public $var = ' 这 里 用 了 类 型 提示 '; 
} 
/* 
下 面 函 数 的 类 型 必须 是 Myclass 类 型 
# 


function MyFunction (MyClass $foo) 
i 

echo $foo->var; 
加 
Smyclass = new MyClass; 
MyFunction($myclass); 
?> 


在 上 述 代 码 中 创建 了 一 个 类 MyClass， 该 类 中 存在 一 个 变量 var。 在 类 外 创建 了 一 个 函数 
MyFunction0， 该 函数 的 参数 是 MyClass 类 的 一 个 对 象 fpo， 此 时 foo 为 形 参 。 在 函数 体 中 ， 直 接 输 
出 了 对 象 foo 的 成 员 变 量 var 的 值 。 然 后 创建 了 该 类 的 实例 化 对 象 myclass， 并 以 此 作为 参数 传递 给 
函数 MyFunction0。 类 型 提示 只 能 够 是 对 象 和 数组 (从 PHP 5.1 开始 ) 类 型 。 传 统 的 类 型 提示 不 支 
持 整 型 和 字符 串 型 。 


5.4.2 ”静态 类 成 员 


声明 一 个 静态 成 员 ， 只 需要 在 方法 前 面 或 者 字段 前 面 加 上 关键 字 static 即 可 。 一 个 类 的 静态 成 
员 会 被 该 类 所 有 的 实例 化 对 象 共 享 ,任何 一 个 实例 化 对 象 对 静态 成 员 的 修改 都 会 映射 到 静态 成 员 中 。 
根据 这 个 性 质 ， 可 以 把 静态 成 员 看 作 是 一 个 全 局 变量 。 静 态 声明 必须 在 可 见 性 声明 之 后 。 为 了 兼容 
PHP 4.0， 如 果 没 有 可 见 性 声明 ， 那 么 成 员 和 方法 将 被 当 作 已 经 声明 为 public。 

类 的 静态 成 员 与 一 般 的 类 成 员 不 同 : 静态 成 员 与 实例 化 对 象 无 关 ， 只 与 类 有 关 。 它 们 用 来 实现 
类 要 封装 的 功能 和 数据 ， 但 不 包括 特定 对 象 的 功能 和 数据 。 静 态 成 员 包括 静态 方法 和 静态 属性 。 

静态 属性 是 包含 在 类 中 要 封装 的 数据 ， 可 以 由 所 有 类 的 实例 化 对 象 共享 。 实 际 上 ， 除 了 属于 一 
个 固定 类 并 限制 访问 方式 外 ， 类 的 静态 属性 非常 类 似 于 函数 的 全 局 变量 。 静 态 属性 不 能 通过 箭头 操 
作 符 -> 访问 。 

静态 方法 则 实现 类 需要 封装 的 功能 ， 与 特定 的 对 象 无 关 。 静 态 方法 非常 类 似 于 全 局 函数 。 静 态 
方法 可 以 完全 访问 类 的 属性 ， 也 可 以 由 对 象 的 实例 来 访问 ， 不 论 访问 的 限定 语 是 什么 关键 字 。 由 于 
静态 方法 可 以 调用 非 对 象 实例 ， 故 伪 变 量 this 不 可 以 在 声明 为 静态 的 方法 中 使 用 。 事 实 上 static 方 
法 的 调用 形式 在 编译 时 被 确定 。 当 使 用 必须 要 声明 的 类 名 时 , 方法 是 完全 标识 和 无 继承 规则 的 应 用 ， 
即 方法 被 完全 确认 ， 而 且 没 有 使 用 继承 规则 。 

下 面 创 建 一 个 案例 ， 演 示 一 下 静态 成 员 的 使 用 。 该 案例 的 代码 如 下 所 示 : 

案例 5-4 

<?php 

class Counter 

private static Scount = 0; 

const VERSION = 2.0; 

function __construct() 

ly 

self::$count++; 

| 

function destruct () 

| 

self: :$count——} 

1 

static function getCount () 

,i 

return self::$count; 

} 

}; 
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$c = new Counter () ; // 创 建 一 个 实例 , 则 _construct () 函数 将 被 执行 
print (" 输 出 的 值 为 : " .Counter: :getCount () . "<br>");// 输 出 1 
print ("类 的 版 本 号 : " . Counter: :VERSION . "<br>") ;// 输 出 类 的 版 本 属性 


?> 


将 上 述 代码 保存 ， 文 件 名 为 staticphp。 打 开 正 浏览 器 ， 在 地 址 栏 中 输入 http://localhost: 
8080/class/static.php， 单 击 【 转 到 】 按 钮 ， 会 显示 如 图 EP 于 T2222 于 富 


雪 社 ) 祷告 区 | 查看 QD 中 窜 必 工具 I] 琴 且 员 


5-4 所 示 的 窗口 。 

在 本 案例 中 创建 了 一 个 私有 的 静态 成 员 count, 一 
个 常量 VERSION 并 赋值 为 2.0。 在 类 的 构造 函数 中 ， 
通过 self 关键 字 ， 调 用 静态 成 员 count， 在 这 里 不 能 用 
this 来 引用 它 ,但 可 以 用 self 或 其 他 有 效 的 命名 表达 式 。 图 54 静态 成 员 
在 该 类 的 构造 函数 中 ,使 用 静态 成 员 执行 自动 加 1， 表 示 每 创建 一 个 实例 化 对 象 ， 该 值 就 加 1。 在 类 
的 析 构 函数 中 ， 是 count 变量 的 值 减 1。 在 静态 方法 getCount0 中 ， 主 要 是 返回 静态 变量 count 的 值 。 
最 后 创建 类 的 对 象 ， 并 调用 相应 的 方法 。 


\ 
0@ | 静态 字段 和 方法 应 使 用 self 关键 字 和 类 名 来 引用 ， 而 不 是 通过 this 和 箭头 操作 符 来 引用 。 


示 


5.4.3 ”instanceof 关键 字 


PHP 5.0 引入 了 instanceof 关键 字 ， 人 允许 用 它 来 测试 一 个 对 象 是 否 是 一 个 类 的 实例 ,或 者 是 一 个 
派生 类 的 实例 ， 或 者 是 某 个 接口 的 实例 。 该 关键 字 的 用 法 简单 ， 其 使 用 示例 如 下 所 示 : 

<?php 

class baseClass { } 

$a = new baseClass; 


if ($a instanceof baseClass) { 


echo "是 该 类 的 实例 化 对 象 "; 
二 


在 上 述 代 码 中 创建 了 一 个 类 baseClass， 该 类 没有 任何 一 个 成 员 。 然 后 创建 了 该 类 的 对 象 a， 使 
用 站 语句 进行 判断 ，a 是 否 是 baseClass 类 的 实例 化 对 象 ， 如 果 是 则 输出 字符 串 信 息 ， 和 否则 不 执行 该 
语句 。 这 里 需要 注意 的 是 类 名 没有 带 界定 符 〈 引 号 )。 如 果 使 用 了 界定 符 将 导致 错误 。 如 果 比 较 失 败 ， 
脚本 将 退出 执行 。instanceof 关键 字 在 同时 处 理 多 个 对 象 时 特别 有 用 。 


5.4.4 ”自动 加 载 对 象 


创建 一 个 大 型 的 PHP 项目， 可 能 需要 创建 大 量 的 类 库 来 完成 指定 的 项 目 ， 从 而 达到 功能 的 相互 
分 离 和 代码 的 重复 使 用 。 在 一 个 PHP 页 面 中 ， 要 使 用 这 些 类 库 或 者 函数 ， 需 要 使 用 require_onceO 
语句 来 实现 。 例 如 ， 如 果 PHP 页 面 中 需要 使 用 一 个 类 ， 就 需要 在 前 面 插入 如 下 语句 : 


require once("myweb/student.class.php"); 


如 果 要 在 一 个 PHP 页 面 中 使 用 多 个 语句 , 采用 这 种 方法 会 变 得 非常 麻烦 。 为 消除 这 个 额外 的 任 
务 ，PHP 5.0 引入 了 自动 加 载 对 象 的 概念 。 自 动 加载 的 实现 ， 需 要 通过 ”_autoload0 函 数 实现 。 

当 尝 试 使 用 一 个 未 定义 的 类 时 ，PHP 会 报告 一 个 致命 错误 。 解决 方法 就 是 添加 一 个 类 ， 可 以 月 
include 包含 一 个 文件 ， 需 要 知道 要 调用 哪个 类 。PHP 提供 了 类 的 自动 加 载 功 能 ， 可 以 节省 编程 的 时 
间 。 当 尝试 使 用 一 个 PHP 没有 组 织 到 的 类 ， 它 会 寻找 一 个 _autoloadO 的 全 局 函数 。 如 果 存 在 这 个 


函数 ，PHP 会 用 一 个 参数 来 调用 它 ， 参 数 即 类 的 名 称 。 其 使 用 示例 如 下 所 示 : 


<?php 
function __autoload( $className) { 

include once S$className . ".php"; 
//require once("classes/$class.class.php"); 
} 
S$object = new ClassName; 
> 


在 上 述 代码 中 ， 函数 中 的 语句 表示 加 载 指 定 的 PHP 页 面 。 当 定义 了 这 个 函数 后 ,就 不 需要 再 使 
用 require_once 语句 或 者 include_once 语句 了 。 


\ 
0 [ __antoloadO 函 数 中 抛 出 的 异常 不 能 被 catch 语句 块 捕获 ， 并 导致 致命 错误 
示 


5.5 “类 /对 象 函数 


PHP 5.0 提供 了 一 些 内 置 函 数 ， 用 来 帮助 开发 人 员 管 理 和 使 用 类 库 。 本 节 将 对 常用 的 内 置 函 数 
进行 详细 的 介绍 。 其 详细 信息 如 表 5-3 所 示 。 
表 5-3 ”类 /对 象 函数 表 


名 称 语法 格式 功 能 


class_exits() bool class_exists (string $class_ name [. bool 检查 类 是 否 已 定义 

$autoload]) 
get_class() string get_class ([object $obj]) 返回 对 象 的 类 名 
get class_ methodsO) array get_class_methods (mixed $class name) 返回 由 类 的 方法 名 组 成 的 数组 
get_class_vars() array get_class_vars (string $class_ name) 返回 由 类 的 默认 属性 组 成 的 数组 
get_declared classes(\ | array get_declared classes (void) 返回 由 已 定义 类 的 名 字 所 组 成 的 数组 
get_ object varsO array get_object_vars (object $obj) 返回 由 对 象 属性 组 成 的 关联 数组 
method_exists|O bool method exists (object $object. string 检查 类 的 方法 是 否 存 在 

$method name) 


1. class_exits() 
如 果 由 class_ name 指定 的 类 已 经 定义 ， 则 返回 Tue， 和 否则 返回 False。 该 函数 的 使 用 示例 如 下 
所 示 : 
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2. get_class() 

该 函数 返回 对 象 实例 obj 所 属 类 的 名 字 。 如 果 obj 不 是 一 个 对 象 则 返回 False。 需 要 注意 的 是 ， 
在 PHP 扩展 库 中 定义 的 类 返回 其 原始 定义 的 名 字 。 在 PHP 4.0 中 get_class0 函 数 返回 用 户 定义 的 类 
名 的 小 写 形式 , 但 是 在 PHP 5.0 中 将 返回 类 名 定义 时 的 名 字 ， 如 同 扩展 库 中 的 类 名 一 样 。 自 PHP 5.0 
起 ， 如 果 在 对 象 的 方法 中 调用 ， 则 obj 为 可 选项 。 该 函数 的 使 用 示例 如 下 所 示 : 


3. get_class_methods() 
该 函数 返回 由 class_name 指定 的 类 中 定义 的 方法 名 所 组 成 的 数组 。 如 果 出 错 ， 则 返回 NULL。 
从 PHP 4.0.6 开始 ， 可 以 指定 对 象 本 身 来 代 蔡 class name。 该 函数 的 使 用 示例 如 下 所 示 : 


在 上 述 代码 中 使 用 了 两 种 方式 给 get _ class methodsO 函 数 传递 参数 。 

4. get_class_vars() 

该 函数 返回 由 类 的 默认 公有 属性 组 成 的 关联 数组 ， 此 数组 的 元 素 以 vamame =>value 的 形式 存 
在 。 注 意 ， 在 PHP 4.0.2.0 之 前 ，get class_vars0) 函 数 不 会 包含 未 初始 化 的 类 变量 。 该 函数 的 使 用 示 
例如 下 所 示 : 

<?php 


class myclass 


| 


public $varl; // 此 变量 没有 默认 值 
public $var2 = "xyz"; 
public $var3 = 100; 
private $var4; 
function __construct() { 
Sthis->val = "foo"; 
$this->va2 = "bar"; 
} 
} 
Smy class = new myclass(); 
$class vars = get class vars (get class (Smy class)); 
foreach ($class vars as $name => $value) { 
echo "$name : $value<br>"; 


2> 

5. get_declared_classes() 

该 函数 返回 由 当前 脚本 中 已 定义 类 的 名 字 组 成 的 数组 。 需 要 注意 的 是 ， 在 PHP 4.0.1pl2 中 ， 有 
3 个 额外 的 类 存在 于 返回 的 数组 中 : stdClass (在 Zend/zend.c 中 定义 )、OverloadedTestClass (在 
ext/standard/basic_functions.c 中 定义 ) 和 Directory (在 ext/standard/dir.c 中 定义 )。 还 需要 注意 的 是 
额外 类 的 出 现 依赖 于 已 编译 到 PHP 中 的 库 , 这 意味 着 不 能 使 用 这 些 类 名 定义 自己 的 类 ， 该 函数 的 使 
用 示例 如 下 所 示 : 

<?php 


print r(get declared classes()); 
2 


6. get_object_vars 

该 函数 返回 由 obj 指定 的 对 象 中 定义 的 属性 组 成 的 关联 数组 。 注意， 在 PHP 4.0.2.0 之 前 的 版 本 
中 , 如 果 在 obj 对 象 实例 中 声明 的 变量 没有 被 赋值 , 则 它们 将 不 会 在 返回 的 数组 中 。 而 在 PHP 4.0.2.0 
之 后 ， 这 些 变量 作为 键 名 将 被 赋予 NULL 值 。 该 函数 的 示例 如 下 所 示 : 


<?php 
class Point2D { 
public $x, S$y; 
public $label; 
function Point2D($x, $y) 
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7. method_exists() 
如 果 method_name 所 指 的 方法 在 object 所 指 的 对 象 类 中 已 定义 ， 则 返回 Tme， 否 则 返回 False。 
该 函数 的 示例 如 下 所 示 : 
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对 于 类 和 对 象 的 其 他 函数 ， 会 在 后 面 的 章节 中 介绍 。 


第 6 章 高 级 OOP 特性 


二 上 学 习 目 标 | objectve 


在 PHP 面向 对 象 的 编程 中 , 不 但 可 以 利用 对 象 封装 的 特性 创建 类 和 对 象 ,实现 程序 数据 的 处 理 
功能 ， 还 可 以 利用 类 的 继承 或 者 接口 ， 实 现 PHP 程序 高 级 的 处 理 功 能 。 本 章 将 对 OOP 的 一 些 高 级 
特性 进行 介绍 ， 如 继承 、 接 口 、 抽 象 类 等 。 


内容 摘要 apstract 


。 了 解 PHP 不 支持 哪些 面向 对 象 的 特性 
。 掌握 在 PHP 对 象 之 间 的 克隆 

。 掌握 clone( 方 法 的 使 用 

。 掌握 在 PHP 中 类 之 间 的 继承 和 使 用 

。 掌握 在 PHP 中 创建 和 使 用 接口 

。 了 解 在 PHP 中 接口 之 间 的 继承 关系 

。 掌握 在 PHP 中 抽象 类 的 创建 和 使 用 

。 掌握 PHP 的 反射 

。 掌握 PHP 对 象 之 间 的 引用 和 比较 


6.1 PHP 不 支持 的 高 级 OOP 特性 


在 PHP 5.0 中 ， 调 整 后 的 面向 对 象 有 些 类 似 于 Java， 如 创建 类 、 实 例 化 对 象 、 构 造 函 数 等 ， 但 
是 不 能 把 Java 的 OOP 的 所 有 特性 都 放 到 PHP 语言 中 。PHP 支持 面向 对 象 的 程序 设计 ， 但 是 它 的 支 
持 还 不 完善 ， 比 如 一 些 高 级 特性 ， 在 PHP 中 就 没有 出 现 。 
为 了 对 PHP 有 一 个 更 加 清晰 的 认识 ， 现 在 把 PHP 中 不 支持 的 高 级 OOP 特性 一 一 列 出 来 ， 如 表 
6-1 所 示 。 
表 6-1 ”PHP 中 不 支持 的 OOP 特 性 
名 称 说 明 


永久 对 象 在 OOP 中 永久 对 象 是 可 以 在 多 个 应 用 的 引用 中 保持 状态 和 功能 的 对 象 ， 这 意味 着 拥有 将 
对 象 保存 到 一 个 文件 或 数据 库 中 的 能 力 ， 而 且 可 以 以 后 装 入 对 象 ， 这 就 是 所 谓 的 序列 化 
机 制 。PHP 拥有 序列 化 方法 ， 它 可 以 通过 对 象 进 行 调用 ， 序 列 化 方法 可 以 返回 对 象 的 字 
符 串 表 示 。 然 而 ， 序 列 化 只 保存 了 对 象 的 成 员 数 据 而 不 包括 方法 

命名 空间 在 PHP 5.0 的 初期 计划 中 ， 准 备 将 命名 空间 〈 包 ) 作为 PHP 的 一 个 特性 ， 但 是 后 来 去 除 


了 对 命名 空间 的 支持 。 实 际 上 命名 空间 主要 是 解决 类 之 间 的 命名 冲突 和 组 织 结构 的 。 相 
信 在 PHP 的 未 来 版 本 中 ， 会 支持 命名 空间 的 特性 
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名 称 说 明 

多 重 继承 继承 的 出 现 , 一 方面 节省 代码 的 编写 , 一 方面 是 使 类 具有 了 一 个 很 好 的 层次 结构 。 在 PHP 
中 支持 单 根 继承 ， 不 支持 多 重 继承 ， 即 一 个 类 只 能 继承 一 个 类 。 有 时 ， 根 据 程序 的 需要 ， 
可 能 要 继承 多 个 对 象 ， 这 时 采用 的 方法 是 继承 多 个 接口 

方法 重 载 方法 重 载 实际 是 类 多 态 的 一 种 表现 形式 ， 即 多 个 方法 具有 相同 的 方法 名 称 ， 但 是 具有 不 
同 的 参数 个 数 和 类 型 。 通 过 名 称 调用 方法 , 参数 决定 执行 的 是 哪个 方法 。 在 PHP 5.0 中 不 
支持 该 特性 。 估 计 将 来 也 不 会 支持 该 特性 

操作 符 重 载 目前 不 支持 根据 所 修改 数据 的 类 型 为 操作 符 赋予 新 的 含义 。 根 据 Zend 网 站 的 讨论 ， 该 特 


性 实现 的 可 能 性 也 不 会 太 大 


对 于 上 面 的 特性 ， 在 PHP 5.0 中 不 会 出 现 ， 在 本 书 的 讲解 中 ， 也 不 会 出 现 这 些 特性 。 


6.2 ”对 象 克隆 


PHP 4.0 没有 提供 一 种 机 制 来 让 用 户 自 定义 复制 构造 子 (copy constructor) 控制 对 象 的 复制 过 程 。 
PHP 4.0 中 所 做 的 二 进 制 复制 ， 能 够 很 精确 地 复制 对 象 的 所 有 属性 。 精 确 地 复制 对 象 的 所 有 属性 可 
能 并 不 是 用 户 需要 的 。 有 个 例子 可 以 很 好 地 说 明确 实 需要 复制 构造 子 : 比如 一 个 GTK Window 的 对 
象 a。a 持 有 它 所 需要 的 全 部 资源 。 当 复制 这 个 GTK Window 到 对 象 b 时 ， 用 户 更 希望 b 持 有 新 的 
资源 对 象 。 再 举 个 例子 : 对象 a 包含 一 个 对 象 c， 当 把 对 象 a 复制 到 对 象 c 时 ， 用 户 可 能 更 希望 对 
象 b 包含 一 个 新 的 对 象 e 的 copy， 而 不 是 一 个 对 象 c 的 引用 。 

换 名 话说， 复制 一 个 对 象 不 仅仅 需要 该 对 象 现 有 的 一 切 属性 ， 如 需要 的 是 复制 的 对 象 的 方法 能 
够 在 新 的 参数 下 进行 操作 ， 得 到 新 的 结果 ， 此 时 复制 过 来 的 是 一 种 新 的 结构 ， 新 的 形态 。 下 面 以 例 
子 的 形式 阐述 一 下 克隆 的 概念 。 


6.2.1 克隆 


PHP 5.0 中 的 对 象 模型 通过 引用 来 调用 对 象 ， 但 有 时 可 能 要 建立 一 个 对 象 的 副本 ， 并 希望 原来 
对 象 的 改变 不 影响 副本 。 为 了 达到 这 样 的 目的 ，PHP 定义 了 一 个 特殊 的 方法 ， 称 为 ”clone。 像 
__construct 和 ”_destruct 一 样 ， 前 面 有 两 个 下 划 线 。 默 认 地 ， 使 用 _clone0 方 法 将 建立 一 个 与 原 对 
象 拥有 相同 属性 和 方法 的 对 象 。 如 果 要 在 克隆 时 改变 默认 的 内 容 ， 需 要 在 。_clone0 方 法 中 重 写 属性 
或 方法 。 克 隆 的 方法 可 以 没有 参数 ， 但 它 同时 包含 this 和 that 指针 that 指向 被 复制 的 对 象 )。 如 果 
选择 克隆 自己 ， 要 小 心 复制 任何 对 象 包含 的 信息 ， 从 that 到 this。 如 果 用 __clone 来 复制 ，PHP 不 会 
执行 任何 隐 性 的 复制 。 

当 一 个 对 象 被 克隆 时 ，PHP 5.0 将 执行 一 个 所 有 对 象 的 属性 的 浅 拷贝 。 任 何 对 其 他 变量 引用 的 
属性 将 只 保留 引用 。 如 果 一 个 __clone0 方 法 被 定义 ， 然 后 重新 创建 一 个 对 象 的 克隆 方法 ， 来 允许 任 
何必 需 的 属性 在 它 需 要 被 改变 时 调用 。 

把 一 个 类 的 对 象 克隆 成 另外 一 个 对 象 ， 语 法 格式 如 下 所 示 : 


$copy_of object = clone S$object; 


在 上 述 代 码 中 ，copy_of object 是 一 个 通过 克隆 产生 的 新 对 象 ，object 是 一 个 被 克隆 的 对 象 。 


下 面 创建 一 个 案例 ， 演 示 一 下 克隆 的 过 程 及 使 用 。 该 案例 的 代码 如 下 所 示 : 


案例 6-1 
<?php 
class foo { 
Private $x; 
private $y; 
function setX(SX) { 
Sthis->x = $x; 
} 
function getX() { 
return $this->x; 
} 
function setY($y) { 
Sthis->y = $y; 
} 
function getY() { 
return S$this->y; 
} 
1 
$01 = new foo; 
Sol->setX(4) 7 
S$ol->setY(8) 7 
$02 =clone $ol7 
$02->setX(5); 
if($01->getXx() != $02->getx()) 
print (" 值 不 相等 ") ; 
echo "<br>ol 对 象 的 值 为 :"; 
echo $01->getxX(); 
echo "gnbsp;&nbsp;"; 
echo $01->getY(); 
echo "<br>o2 对 象 的 值 为 : "; 
echo $02->getX(); 
echo "gnbsp;&nbsp;"; 
echo $02->getY(); 
?> 


将 上 述 代 码 保存 ， 文 件 名 为 clone.php， 该 文件 保存 到 C:\Web\apache\htdocs\class 目录 下 ， 本 章 
的 案例 都 是 保存 在 该 目录 下 ， 以 后 不 再 特别 声明 。 打开 下 EECTTECESS GD 
浏览 器 ， 在 地 址 栏 中 输入 http://localhost:8080/class/clone. 让 刷 图 辆 
php， 单 击 【 转 到 】 按 钮 ， 会 显示 如 图 6-1 所 示 的 窗口 。 

在 上 述 代码 中 ， 首 先 创 建 一 个 类 foo， 在 该 类 中 创建 
了 两 个 变量 x、y， 并 为 这 两 个 变量 分 别 创建 了 相应 的 设置 
变量 值 和 获取 变量 值 的 方法 ， 如 gefXO 和 getY0 方 法 。 在 
类 的 下 面 使 用 “$ol= new foo” 语 句 ， 创 建 foo 类 的 对 象 
ol1， 并 利用 该 对 象 给 变量 x、y 分 别 赋值 4 和 8。 赋 值 完 成 后 ， 利 用 语句 “$o2 -clone $o1 ”克隆 一 个 
新 的 对 象 2， 此 时 对 象 02 就 具有 了 对 象 ol 现在 所 具有 的 属性 值 ， 如 对 象 o2 中 的 变量 x、y 的 值 ， 
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并 且 该 对 象 还 具有 对 象 ol 中 的 方法 。 在 下 面 利用 对 象 o2 中 的 方法 setX0 设 置 变 量 x 的 值 ， 这 时 变 
量 x 的 值 发生 了 变化 ,但 是 变量 y 的 值 没有 发 生变 化 ,还 是 原来 的 值 .“if($01->getX0 (= $02->getX0)” 
表示 对 变量 x 的 值 进行 比较 ， 看 二 者 是 否 相等 ， 如 果 不 相等 ,执行 输出 语句 ,否则 放弃 执行 。 最 后 ， 
分 别 输出 变量 x 和 y 的 值 。 

从 本 案例 中 可 以 看 出 ， 当 克隆 一 个 新 对 象 时 ， 新 对 象 不 但 具有 原来 对 象 的 属性 ， 还 可 以 通过 克 
隆 来 的 方法 重新 对 变量 的 值 进 行 设 定 。 


6.2.2 __clone() 方 法 


在 对 象 克 隆 期 间 ， 有 时 不 希望 把 所 有 的 属性 都 克隆 过 来 ， 希 望 有 些 局 部 位 置 做 一 点 改动 ， 如 一 


class MyCloneable { 
public static $id = 0; 
function construct() { 
$this->id = self::S$Sid++7 
3 
function __clone() { 
$this->address = "New York"; 
$this->id = self::$id++; 
} 
1 
$0obj = new MyCloneable(); 
$obj->name = "Hello"; 
$obj->address = "Tel-Aviv"; 
Drint Sobij= id . TNnny 
S$obj_cloned = clone $obj; 
print $0obj cloned->id . "\n"; 
print $obj cloned->name . "\n™; 
print $obj cloned->address . "\n"™; 
EE 


将 上 述 代码 保存 ， 文 件 名 为 clonel.php， 并 保存 到 指定 的 位 置 。 打 开 正 浏览 器 ， 在 地 址 栏 中 输 
入 http://localhost:8080/class/clonel.php, 单 击 【 转 到 】 按 钮 ， 
会 显示 如 图 6-2 所 示 的 窗口 。 rr 

在 上 述 代码 中 创建 了 一 个 类 MyCloneable， 并 为 该 类 
创建 了 一 个 静态 变量 、 一 个 构造 函数 、 一 个 __clone0 方 法 。 
在 构造 函数 中 ， 设 置 静态 变量 id 的 值 ， 每 调用 一 次 就 增加 
一 次 。 在 __clone( 方 法 中 ， 设 置 在 进行 克隆 对 象 后 ， 还 需 图 62 __clone 案例 


卫 

个 字段 的 值 。 这 时 可 以 采用 在 克隆 的 类 中 定义 一 个 方法 __clone0， 从 而 调整 对 象 的 克隆 行为 ， 即 在 
三 克隆 的 过 程 中 ， 可 以 将 一 些 局 部 的 代码 设置 成 自己 需要 的 代码 。 其 执行 是 将 现 有 的 对 象 成 员 复制 到 
风 目标 对 象 之 后 ， 还 会 执行 __clone0 方 法 指定 的 操作 。 

a 下 面 创建 一 个 案例 ， 演 示 __clone0 方 法 的 使 用 。 该 案例 的 代码 如 下 所 示 : 
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要 调整 的 代码 ， 如 地 址 信息 为 “New York” 变量 id 的 值 在 原来 的 基础 上 加 1。 使 用 语句 “$obj =new 
MYyCloneable0 ”创建 一 个 对 象 obj， 并 利用 该 对 象 设置 变量 name 和 address 的 值 ， 并 输出 变量 id 的 
值 。 使 用 语句 “$obj_cloned = clone $obj” 产 生 一 个 新 的 克隆 对 象 ， 这 时 该 对 象 具 有 obj 的 属性 和 方 
法 ， 然 后 输出 克隆 的 对 象 obj_cloned 此 时 具有 的 变量 值 ， 这 时 变量 id 的 值 变 化 了 ， 变 量 address 的 
值 不 再 是 前 面 设 定 的 “Tel-Aviv” 了 ， 而 是 在 。”_clone0 方 法 中 设置 的 “New York” 信 息 。 

从 本 案例 中 可 以 看 出 ，__clone0 方 法 可 以 修改 克隆 对 象 中 的 一 些 代码 。 


6.3 ”继承 


继承 是 OOP 程序 设计 语言 的 基本 特征 ， 所 有 的 面向 对 象 的 语言 都 支持 该 特性 。 继 承 的 产生 使 
类 和 类 之 间 有 了 相应 的 层次 结构 ， 使 类 的 管理 更 加 清晰 ， 并 且 提 高 了 代码 的 可 重用 性 ， 如 一 些 功能 
可 以 通过 继承 获得 。 在 PHP 中 ， 可 以 实现 一 个 类 继承 另外 一 个 类 ， 但 是 PHP 不 支持 多 重 继承 ， 即 
继承 多 个 类 。 本 节 将 对 类 的 继承 进行 详细 的 介绍 。 


6.3.1 类 继承 


在 PHP 中 , 类 和 类 之 间 属 于 单 根 继承 , 当 一 个 类 继承 另外 一 个 类 , 该 类 被 称 为 子 类 (child class)， 
被 继承 的 类 称 为 父 类 (parent class)， 此 外 ， 子 类 可 以 被 称 为 超 类 ， 父 类 可 以 被 称 为 基 类 。 当 然 ， 一 
个 类 可 以 被 多 个 子 类 来 继承 ， 一 个 类 只 能 拥有 一 个 父 类 。 类 继承 的 语法 格式 如 下 所 示 : 


class subclass extends superclass{ 


方法 体 
} 


在 上 述 代码 中 ，subclass 表示 子 类 ，extends 表示 类 的 继承 符 ，superclass 表示 父 类 。 子 类 不 但 可 
以 拥有 父 类 的 成 员 ， 如 方法 和 字段 ， 还 可 以 拥有 自己 本 身 新 增 的 方法 。 在 继承 过 程 中 ， 子 类 不 能 拥 
有 父 类 的 私有 成 员 。 

下 面 创 建 一 个 案例 ， 演 示 类 的 继承 。 该 案例 的 代码 如 下 所 示 : 


案例 6-3 
<?php 
class MyClass { 
private S$Hello = "Hello, World!\n"; 
public $Bar = "Hello, Foo!\n™; 
public SFoo = "Hello, Bar!\n™; 
function printHello() { 
print "<br>MyClass::printHello() " . $this->Hello; 
print "<br>MyClass::printHello() " . $this->Bar; 
print "<br>MyClass::printHello() " . $this->Foo; 
} 
1 
class MyClass2 extends MyClass { 
public $f=" 大 家 好 "; 
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function printWorld(){ 
if($Bar="Hello, Foo!\n") 
{echo "<br> 子 类 的 判断 <br>";} 
} 
1 
S$obj = new MyClass(); 
S$obj->printHello(); 
S$objl = new MyClass2(); 
S$obj1->printHello(); 
S$obj1->printWorld(); 
print $objl->Foo; 
print S$objl->f7 
?> 


将 上 述 代码 保存 ， 文 件 名 为 extends.php， 并 保存 到 指定 的 位 置 。 打 开 正 浏览 器 ， 在 地 址 栏 中 
输入 http://localhost:8080/class/extends.php, 单 击 【 转 ge 
到 】 按 钮 ， 会 显示 如 图 6-3 所 示 的 窗口 。 

在 上 述 代码 中 创建 了 一 个 类 MyClass, 在 该 类 中 
创建 了 三 个 变量 ， 分 别 为 Hello、Bar、Foo， 并 分 别 
赋值 。 在 方法 printHello0 中 ， 输 出 三 个 变量 的 值 。 
然后 创建 一 个 类 MyClass2 继承 MyClass， 并 在 
MyClass2 中 新 建 了 一 个 方法 printWorld0， 这 时 类 
MyClass2 具有 两 个 方法 ，printHello0 和 printWorldO 
方法 。 在 类 MyClass2 中 又 新 建 了 一 个 变量 f 并 赋值 。 类 创建 完成 之 后 ， 在 下 面 创建 两 个 类 的 实例 化 
对 象 obj、objl1， 并 利用 对 象 调用 相关 的 方法 和 成 员 变 量 。 

从 本 案例 的 代码 中 可 以 看 出 ， 一 个 子 类 可 以 继承 父 类 的 方法 和 字段 ， 并 且 可 以 创建 属于 自己 的 
方法 和 字段 ， 并 且 在 子 类 中 也 可 以 重 写 父 类 的 方法 。 


6.3.2 ”继承 和 构造 函数 


在 上 一 章 中 介绍 了 一 个 类 可 以 调用 另外 一 个 类 的 构造 函数 。 如 果 一 个 子 类 继承 了 一 个 父 类 ， 子 
类 应 该 可 以 调用 父 类 的 构造 函数 。 在 子 类 的 构造 函数 中 调用 父 类 的 构造 函数 有 两 种 方式 ， 第 一 种 方 
式 如 下 所 示 : 

classname:: _construct ($name) 

这 种 方式 和 前 面 的 相同 ， 类 名 加 上 构造 函数 名 。 该 种 方式 称 为 显 式 调用 。 第 二 种 调用 方式 如 下 
所 示 : 

parent:: _construct(); 

在 上 述 代码 中 ，parent 表示 父 类 ， 关 键 字 加 上 构造 函数 名 。 这 种 方式 称 为 隐 式 调用 。 下 面 创建 
一 个 案例 ， 演 示 在 子 类 中 调用 父 类 的 构造 函数 。 该 案例 的 代码 如 下 所 示 : 

案例 6-4 

<?php 


图 6-3 类 继承 


class BaseClass { 
function construct() { 
print " 父 类 的 构造 函数 <br>"; 
} 
class SubClass extends BaseClass { 
function construct() { 
BaseClass:: construct(); 
parent:: construct(); 
print " 子 类 的 构造 函数 <br>"; 
} 
上 
$0obj = new BaseClass(); 
S$obj = new SubClass(); 


2> 
将 上 述 代码 保存 ， 文 件 名 为 construct.php， 并 保存 到 指定 的 位 置 。 打 开 正 浏览 器 ， 在 地 址 栏 中 
输入 http://localhost:8080/class/construct.php, 单 击 【 转 到 】 [ET 于 于 
按钮 ， 会 显示 如 图 6-4 所 示 的 窗口 。 te 


在 本 案例 中 创建 了 两 个 类 ， 分别 为 BaseClass 和 
SubClass， 在 父 类 BaseClass 中 的 构造 函数 中 输出 一 个 字 
符 串 信息 ， 在 子 类 SubClass 中 调用 父 类 的 构造 函数 ， 采 
用 了 两 种 方式 ， 分 别 为 BaseClass:: _constructO、parent:: 
__construct0， 并 在 子 类 的 构造 函数 中 ， 输 出 一 个 字符 串 
信息 。 


6.4 接口 


众所周知 ，PHP 4.0 中 的 对 象 支持 继承 ， 要 使 一 个 对 象 成 为 男 一 个 对 象 的 派生 类 ， 需 要 使 用 类 
似 于 “class sub extends parent” 的 代码 来 控制 。 在 PHP 4.0 和 PHP 5.0 中 ， 一 个 对 象 仅 能 继承 一 次 ， 
多 重 继 承 是 不 被 支持 的 。 不 过 ， 在 PHP 5.0 中 产生 了 一 个 新 的 名 词 : 接口， 接口 是 一 个 没有 具体 处 
理 代码 的 特殊 对 象 , 它 仅 仅 定义 了 一 些 方法 的 名 称 及 参数 。 本 节 将 会 详细 地 介绍 接口 的 创建 和 使 用 。 


6.4.1 ”实现 一 个 接口 


对 象 接口 允许 创建 一 个 指定 类 方法 的 执行 代码 ,而 不 必 说 明 这 些 方法 是 如 何 被 操作 (处理) 的 。 
接口 作为 一 个 标准 类 ， 没 有 任何 方法 有 它们 内 容 的 定义 。 在 接口 中 所 有 的 方法 必须 声明 为 public， 
这 是 接口 的 特性 。 创 建 一 个 接口 需要 使 用 关键 字 interface。 

一 个 接口 的 语法 格式 如 下 所 示 : 


interface myinterface{ 
const namel; 
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在 上 述 代码 中 ，interface 表示 创建 一 个 接口 ，myinterface 表示 接口 的 名 称 。namel 等 表示 在 接 
口中 声明 的 字段 ， 是 一 个 常量 。methodNamel 表示 方法 的 名 称 ， 该 方法 没有 方法 体 。 


0 通常 ， 在 接口 名 称 前 面 加 上 字母 I 来 进行 标识 ， 以 便 更 容易 辩 认 。 


接口 的 使 用 意义 在 于 定义 了 一 系列 的 标准 ， 让 别 的 类 去 实现 ， 接 口 的 使 用 使 类 之 间 的 关系 产生 
层次 感 。 如 果 创建 了 一 个 接口 ， 而 没有 相关 的 类 去 实现 ， 该 接口 是 毫 无 意义 的 。 接 口 需要 继承 来 实 
现 它 的 价值 。 一 个 类 继承 一 个 接口 的 语法 格式 如 下 所 示 : 


在 上 述 代码 中 ，implements 关键 字 表示 实现 〈 执 行 ) 一 个 接口 。 在 接口 中 所 有 的 方法 必须 在 一 
个 类 的 内 部 实现 ， 朴 忽 这 些 将 导致 一 个 致命 错误 。 类 可 以 实现 多 个 接口 。 通 过 使 用 一 个 逗号 分 开 每 
个 接口 。 

下 面 创 建 一 个 案例 ， 演 示 一 下 如 何 创建 接口 ， 并 实现 接口 中 的 方法 。 该 案例 的 代码 如 下 所 示 : 


2 A(Q); 
$a->getVv(); 
$a->getH ("<br> 郑 州 ", "欢迎 你 ") ; 
?> 
将 上 述 代码 保存 ， 文 件 名 为 interfacephp， 并 将 文件 保存 到 指定 的 位 置 。 打 开 正 浏览 器 ， 在 地 
址 栏 中 输入 http://localhost:8080/class/interface.php, 单 击 pz 8060 rozott 工 .后 司 民 | 
【 转 到 】 按 钮 ， 会 显示 如 图 6-5 所 示 的 窗口 。 1 
在 上 述 代 码 中 创建 了 一 个 接口 myinterface， 在 接口 
中 定义 了 一 个 常量 和 两 个 方法 。 这 些 方法 默认 为 公共 的 
作用 域 ， 并 没有 相应 的 方法 体 ， 只 有 方法 的 名 称 。 在 下 
面 创建 一 个 类 A 继承 了 该 接口 ， 并 实现 类 接口 中 的 方法 国 69 于 中 实 多 
getVO 和 getHO。 在 下 面 创建 了 A 类 的 对 象 a， 利 用 该 对 象 调用 实现 了 的 方法 。 


6.4.2 ”实现 多 个 接口 


一 个 类 不 仅 可 以 实现 一 个 接口 ， 也 可 以 同时 实现 多 个 接口 。 当 实现 多 个 接口 时 ， 接 口 之 间 用 逗 
号 隔 开 。 其 使 用 示例 如 下 所 示 : 


<?php 
interface displayable { 
function display(); 
0 
interface printable { 
function doprint (); 
1 
class foo implements displayable,printable { 
function display() { 
echo "天 气 晴 朗 ， 万 里 无 云 "; 
} 
function doprint() { 
echo "出 去 走 一 走 "; 
} 
1 
$f=new foo(); 
$f->display(); 
$f->doprint (); 
> 


在 上 述 代 码 中 创建 了 一 个 接口 displayable， 并 在 接口 中 定义 了 一 个 方法 display0。 然 后 创建 了 
另外 一 个 接口 printable, 在 接口 中 定义 了 一 个 方法 doprint0。 在 下 面 创建 了 一 个 类 foo 继承 了 上 面 两 
个 接口 ， 并 实现 接口 中 的 方法 。 然 后 创建 foo 类 的 对 象 f， 利用 该 对 象 调用 已 实现 的 display0 方 法 和 
doprint0 方 法 。 

接口 实际 上 可 以 看 作 一 个 类 ， 类 和 类 之 间 可 以 继承 ， 其 实在 PHP 中 , 一 个 接口 可 以 继承 另外 一 
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个 接口 。 其 使 用 示例 如 下 所 示 : 
<2php 


interface Foo { 
public function doFoo(); 
1 
interface Bar extends Foo { 
public function doBar(); 
class Zip implements Bar { 
public function doFoo() { 
echo "Foo"; 
} 
public function doBar() { 
echo "Bar™; 
} 
$zip = new Zip(); 
$2zip->doFoo (); 
$zip->doBar (); 
2 


在 上 述 代码 中 创建 了 一 个 接口 Bar 继承 另外 一 个 接口 Foo， 那 么 此 时 Bar 接口 中 存在 了 两 个 方 
法 doFoo0 和 doBar0。 需 要 注意 的 是 ， 一 个 接口 继承 另外 一 个 接口 需要 使 用 关键 字 extends。 


6.5 ”抽象 类 


面向 对 象 程序 通过 类 的 分 层 结构 构建 起 来 ， 在 单 重 继承 语言 如 PHP 中 ， 类 的 继承 是 树 状 的 。 一 
个 根 类 有 一 个 或 多 个 子 类 ,再 从 每 个 子 类 继承 出 一 个 或 多 个 下 一 级 子 类 。 当 然 , 可 能 存在 多 个 根 类 ， 
用 来 实现 不 同 的 功能 。 在 一 个 良好 设计 的 体系 中 ， 每 个 根 类 都 应 该 有 一 个 有 用 的 接口 ， 可 以 被 应 用 
代码 所 使 用 。 如 果 应 用 代码 被 设计 成 与 根 类 一 起 工作 ， 那 么 它 也 可 以 和 任何 一 个 从 根 类 继承 出 来 的 
子 类 合作 。 根 类 通常 被 设计 为 抽象 类 或 者 接口 。 

抽象 方法 就 是 像 子 类 中 一 般 方法 的 占 位 符 《〈 占 个 地 方 但 不 起 作用 )， 它 与 一 般 方法 不 同 ， 没 有 
任何 代码 。 如 果 类 中 存在 一 个 或 多 个 抽象 方法 ， 那 么 这 个 类 就 成 了 抽象 类 。 该 类 不 能 被 实例 化 抽象 
类 ， 只 允许 继承 它们 ， 然 后 实例 化 子 类 ， 也 可 以 把 抽象 类 看 成 是 子 类 的 一 个 模板 。 

如 果实 现 了 所 有 的 抽象 方法 ， 子 类 就 变 成 一 个 普通 的 类 。 如 果 没 有 覆 写 所 有 方法 ， 子 类 仍 是 抽 
象 的 。 如 果 一 个 类 中 包含 抽象 方法 (哪怕 只 有 一 个 )， 必 须 声明 这 个 类 是 抽象 的 ， 在 class 关键 字 前 
加 上 abstract。 如 果 建 立 了 一 个 只 有 抽象 方法 的 类 ， 就 定义 了 一 个 接口 (interface)。 

下 面 创 建 一 个 案例 ， 演 示 一 下 抽象 类 的 创建 和 继承 。 该 案例 的 代码 如 下 所 示 : 

案例 6-6 

<?php 

abstract class AbstractClass 


abstract protected function getValue () 7 
abstract protected function prefixValue ($prefix); 
public function printout () 
{ 
print Sthis->getValue () ."<br>"7 
是 
1 
class ConcreteClass1l extends AbstractClass 
{ protected function getValue () 
return "<br> 这 是 子 类 1 实现 的 "; 
} 
public function prefixValue ($prefix) 
{ 
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return "<br>{$prefix} 这 是 子 类 1 实现 的 "; 
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class ConcreteClass2 extends AbstractClass 
{ public function getValue () 
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return "<br> 这 是 子 类 2 实现 的 "; 


} 
public function prefixValue ($prefix) 


{ 
return"<br>{$prefix} 这 是 子 类 2 实现 的 "; 
} 


$classl = new ConcreteClass17 

$classl->printOut (); 

echo $classl->prefixValue('FOO ') ."<br>"; 

$class2 = new ConcreteClass2; 

$class2->printOut (); 

echo $class2->prefixValue('FOO ') ."<br>"; 

2 

将 上 述 代码 保存 ， 文 件 名 为 abstractphp， 并 保存 到 指定 的 位 置 。 打 开 正 浏览 器 ， 在 地 址 栏 中 
输入 http://localhost:8080/class/abstract.php， 单 击 回 
【 转 到 】 按 钮 ， 会 显示 如 图 6-6 所 示 的 窗口 。 

在 上 述 代码 中 创建 了 一 个 抽象 类 Abstract- 
Class, 该 类 的 前 面 加 上 关键 字 abstract 表明 该 类 为 
抽象 类 。 在 AbstractClass 中 ， 定 义 了 抽象 方法 
getvalue0 和 带 有 参数 的 prefixgValue0 抽 象 方法 , 还 
创建 了 一 个 普通 的 方法 printOut0 方 法 ， 该 方法 输 
出 一 个 字符 串 信息 。 创 建 另 外 两 个 类 Concrete 
Classl 和 ConcreteClass2 继承 上 面 的 抽象 类 ， 这 两 个 类 分 别 实 现 了 抽象 类 中 的 方法 。 抽 象 类 不 能 被 
直接 实例 化 ， 可 以 通过 实例 化 抽象 类 的 子 类 调用 相应 的 方法 。 

可 以 看 出 ， 抽 象 类 和 接口 比较 相似 ， 这 两 个 对 象 中 都 包含 了 抽象 方法 。 只 不 过 接口 中 全 部 是 抽 
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象 方法 ， 抽 象 类 中 人 允许 带 有 普通 方法 。 在 OOP 的 程序 设计 中 , 什么 样 的 情况 使 用 抽象 类 ， 什 么 样 的 
情况 使 用 接口 ， 可 以 通过 下 列 因 素来 决定 : 
。 如 果 创 建 一 个 模型 ， 这 个 模型 将 由 一 些 紧密 相关 的 对 象 采用 ， 就 可 以 使 用 抽象 类 。 如 果 创 建 
由 一 些 不 相关 对 象 采用 的 功能 ， 就 使 用 接口 。 
。 如 果 必 须 从 多 个 来 源 继承 行为 ， 就 使 用 接口 。PHP 类 可 以 继承 多 个 接口 ， 但 不 能 继承 多 个 抽 
象 类 。 
。 如 果 知 道 所 有 类 都 会 共享 一 个 公共 的 行为 实现 ， 就 使 用 抽象 类 ， 并 在 其 中 实现 该 行为 。 在 接 
口中 无 法 实现 行为 。 


开发 一 个 PHP 页 面 程序 ， 如 网 上 购物 等 ， 可 能 需要 多 个 类 协助 来 完成 。 一 个 这 样 的 程序 中 包含 
十 几 个 类 的 情况 ， 并 不 少见 ， 并 且 每 个 类 中 可 能 会 包含 大 量 的 成 员 和 复杂 的 方法 。 如 果 要 修改 革 
一 个 参数 或 者 查看 某 一 个 方法 的 作用 域 范围 ， 可 能 需要 打开 多 个 PHP 页 面 ， 并 通过 编辑 器 查看 这 些 
信息 。 

通过 查看 一 个 对 象 的 信息 ， 来 把 握 该 对 象 的 特性 的 思想 称 为 自省 ， 进 行 查看 的 过 程 称 为 反射 。 
在 PHP 5.0 中 ， 提 供 了 一 个 反射 API， 不 仅 能 够 查看 类 和 方法 ， 还 能 够 查看 函数 、 接 口 和 子 类 。 反 
射 不 仅仅 用 来 获取 信息 ， 还 可 以 测试 文档 ， 生 成 一 些 帮助 文档 等 。 本 节 将 一 一 介绍 这 些 API 反射 。 


6.6.1 编写 ReflectionClass 类 


ReflectionClass 类 主要 用 于 了 解 一 个 类 的 信息 ， 如 该 类 的 父 类 是 哪个 ， 该 类 具有 哪些 成 员 ， 具 
有 哪些 方法 , 类 的 名 称 , 该 类 是 否 为 抽象 类 , 是 否 是 最 终 类 等 。 ReflectionClass 类 继承 接口 Reflector， 
在 该 类 中 封装 和 实现 了 一 些 相应 的 方法 ， 用 来 获得 要 反射 的 类 的 基本 信息 。 

ReflectionClass 类 中 常用 的 方法 如 下 所 示 : 


<?php 
class ReflectionClass implements Reflector 
{ final Private _ _clone() 
public object construct (string name) 
public string toString () 
public static string export() 
public string getName () 
public bool isInternal() 
public bool isUserDefined() 
public bool isInstantiable() 
public bool hasConstant (String name) 
public bool hasMethod (string name) 
public bool hasProperty(string name) 
public string getFileName() 
public int getStartLine() 
public int getEndLine() 


public string getDocComment () 

public ReflectionMethod getConstructor () 

public ReflectionMethod getMethod (string name) 
public ReflectionMethod[] getMethods () 

public ReflectionProperty getProperty(string name) 
public ReflectionProperty[] getProperties () 
public array getConstants () 

public mixed getConstant (string name) 

public ReflectionClass[] getInterfaces () 

public bool isInterface() 

public bool isAbstract() 

public bool isFinal() 

public int getModifiers () 

public bool isInstance(stdclass object) 

public stdclass newInstance (mixed* args) 

public ReflectionClass getParentClass () 

public bool isSubclassOf (ReflectionClass class) 
public array getStaticProperties () 

public mixed getStaticPropertyValue (string name [, mixed default]) 
public void setStaticPropertyValue (string name, mixed value) 
public array getDefaultProperties () 

public bool isIterateable() 

public bool implementsInterface (string name) 
public ReflectionExtension getExtension() 

public string getExtensionName () 
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对 于 上 面 的 方法 这 里 不 再 一 一 解释 ， 读 者 可 以 根据 方法 的 名 称 来 判读 方法 的 功能 。 为 了 内 省 一 
个 类 ， 必 须 首先 创建 ReflectionClass 类 的 一 个 实例 ， 可 以 随后 访问 这 个 实例 的 任何 上 述 方法 。 
下 面 创建 一 个 案例 ， 演 示 一 下 如 何 获 得 类 的 基本 信息 。 该 案例 的 代码 如 下 所 示 : 


案例 6-7 
<?php 
class Counter 
{ Const START = 07 
private static $c = Counter::START; 
public function count() { return self::$c++; } 
1 
// 创建 一 个 反射 类 的 实例 化 对 象 
$class = new ReflectionCclass ('Counter') 7 
// 输 出 该 类 的 基本 信息 
$methods=$class->getMethods () 7 
echo "该 类 具有 的 方法 有 <br>"; 
foreach ($methods as Smethod) 
echo $method->getName()."<br>"; 
$isAbstract=$class->isAbstract ()?" 是 一 个 抽象 类 " : "不 是 一 个 抽象 类 "; 
$isFinal=$class->isFinal ()?" 是 一 个 最 终 类 ": "不 是 一 个 最 终 类 "7 


echo "<br>"; 


131 


卫 
a 
卫 
十 

= 
< 

op) 
QO 
Pe 
一 
TU 
宇 
学 
习 
手 
册 


132 


echo $class->getName() ." 类 ".$isAbstract; 
echo "<br>"; 

echo $class->getName () ." 类 ".$isFinal; 

?> 


将 上 述 代码 保存 ， 文 件 名 为 Relassphp， 并 保存 到 指定 的 位 置 。 打 开 正 浏览 器 ， 在 地 址 栏 中 输 
入 http://localhost:8080/class/Rcelass.php， 单 击 【 转 
到 】 按 钮 ， 会 显示 如 图 6-7 所 示 的 窗口 。 

在 上 述 代码 中 创建 了 一 个 类 Counter, 在 类 中 
创建 了 一 个 常量 、 一 个 静态 变量 、 一 个 普通 方法 。 
使 用 语句 “$class =new ReflectionClass(CCounter)” 
创建 了 反射 类 的 实例 化 对 象 class, 利用 class 对 象 
并 结合 getMethods() 方 法 获取 类 中 的 方法 ， 结 合 
getName() 方 法 获取 类 的 名 称 ， 结 合 isAbstract0 判 断 该 类 是 否 为 抽象 类 ， 结 合 isFinal0 判 断 该 类 是 否 
为 最 终 类 等 。 


图 6-7 类 反射 


6.6.2 ”编写 ReflectionMethod 类 


ReflectionMethod 类 主要 用 于 获取 方法 的 详细 信息 ， 如 该 方法 是 否 为 抽象 方法 ， 是 否 为 最 终 方 
法 ， 是 否 为 静态 方法 ， 其 作用 域 是 什么 等 。ReflectionMethod 类 包含 的 常用 方法 如 下 所 示 : 


<?php 
class ReflectionMethod extends ReflectionFunction 
{ public construct (mixed class, string name) 
public string tostring() 
public static string export() 
public mixed invoke (stdclass object, mixed* args) 
public mixed invokeArgs (stdclass object, array args) 
public bool isFinal() 
public bool isAbstract() 
public bool isPublic() 
public bool isPrivate() 
public bool isProtected() 
public bool isStatic() 
public bool isConstructor () 
public bool isDestructor() 
public int getModifiers () 
public ReflectionClass getDeclaringClass() 
// Inherited from ReflectionFunction 
final private clone() 
public string getName () 
public bool isInternal () 
public bool isUserDefined() 
public string getFileName() 
public int getstartLine() 
public int getEndLine() 


public string getDocComment () 

public array getstaticVariables() 

public bool returnsReference () 

public ReflectionParameter[] getParameters () 
public int getNumberOfParameters () 

public int getNumberOfRequiredParameters () 


| 

2 

为 了 内 省 一 个 方法 ， 必 须 首 先 创建 ReflectionMethod 类 的 一 个 实例 ， 可 以 随后 访问 这 个 实例 的 
任何 上 述 方法 。 

下 面 创建 一 个 案例 ， 演 示 一 下 如 何 获 取 一 个 方法 的 详细 信息 。 该 案例 的 代码 如 下 所 示 : 

案例 6-8 

<?php 


class Counter 
{ private static $c = 0; 
final public static function increment (){ return ++self::$c; } 


} 
// 以 指定 参数 创建 类 ReflectionMethod 的 实例 化 对 象 
Smethod = new ReflectionMethod('Counter','increment'); 
printf( "===> 这 个 $s%s%s%s%s%s%s 方法 '%s' (这 个 方法 $s) <br>"”. 
声明 在 %s<br>" . 
"” ， 行 数 从 sd to sd<br>" . 
具有 的 修饰 符 为 sd[$s]<br>"， 
Smethod->isInternal() ? "系统 内 置 的 ' : ' 用 户 自 定义 的 '， 
$method->isAbstract () ? ' 抽象 ' : ' 普 通 '， 
Smethod->isFinal() ? ' 最 终 ' : ' 普 通 '， 
Smethod->isPublic() ?公共 ' : ''， 
$method->isPrivate() ? ' 私有 的 ' : '…， 
$method->isProtected() ? ' 受 保护 的 ' : '…， 
Smethod->isStatic() ? ' 静态 的 : ''， 
Smethod->getName () ， 
Smethod->isConstructor () ? "构造 函数 ' : ' 普 通 方法 '， 
$method->getFileName ()， 
Smethod->getStartLine() 
$method->getEndline(), 
$method->getModifiers(), 
implode(' ', Reflection::getModifierNames ($method->getModifiers())) 
Ds 


2> 
将 上 述 代码 保存 ， 文 件 名 为 Rmethod.php, 并 保 此 二 ht 
存 到 指定 的 位 置 。 打 开 正 浏览 器 ， 在 地 址 栏 中 输入 | 王 | 


公共 袁术 的 方法 “inerenent [这 十 方法 普通 方 


http://localhost:8080/class/Rmethod.php， 单 击 【 转 到 】 
按钮 ， 会 显示 如 图 6-8 所 示 的 窗口 。 

本 案例 的 代码 过 于 简单 , 这 里 不 再 一 一 介绍 , 不 
懂 的 地 方 可 以 查看 中 文 PHP 手册 。 图 6-8 方法 反射 
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6.6.3 ”编写 ReflectionParameter 类 


ReflectionParameter 类 主要 用 于 取 回 一 个 函数 或 方法 的 参数 的 信息 , 如 参数 的 名 称 。 该 类 所 具有 
的 方法 如 下 所 示 : 


为 了 内 省 函数 参数 ， 必 须 首 先 创建 ReflectionFunction 或 ReflectionMethod 类 的 一 个 实例 ， 然 后 
用 getParameters0 方 法 返回 一 个 数组 型 参数 。ReflectionParameter 类 的 使 用 示例 如 下 所 示 : 


6.6.4 ”编写 ReflectionProperty 类 


ReflectionProperty 类 人 允许 反 向 设计 类 属性 ， 即 获取 类 的 属性 的 相关 信息 ， 如 属性 的 名 称 、 数 值 、 
作用 域 等 。 该 类 具有 的 方法 如 下 所 示 : 


为 了 内 省 一 个 属性 ， 必 须 首先 创建 ReflectionProperty 类 的 一 个 实例 ， 可 以 随后 访问 这 个 实例 的 
任何 上 述 方法 。 该 类 的 使 用 示例 如 下 所 示 : 


巴山 吕 弟 
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这 里 需要 注意 的 是 ， 试 图 获得 或 设置 私有 的 或 保护 的 类 属性 的 值 将 导致 抛 出 异常 。 


6.6.5 ”编写 ReflectionExtension 类 


ReflectionExtension 类 允许 反 向 设计 扩展 ,可 以 在 使 用 get_loaded_extensions() 函 数 时 重新 返回 所 
有 加 载 的 扩展 ， 即 获得 该 类 在 运行 时 被 加 载 类 的 信息 。 该 类 所 具有 的 方法 如 下 所 示 : 


为 了 内 省 一 个 扩展 ， 必 须 首先 创建 ReflectionExtension 类 的 一 个 实例 。 可 以 随后 访问 这 个 实例 
的 任何 上 述 方法 。 该 类 的 使 用 示例 如 下 所 示 : 


6.7 对 象 的 引用 


在 PHP 4.0 中 ， 给 一 个 函数 或 方法 传递 变量 ， 实 际 上 是 把 这 个 变量 做 了 一 次 复制 ， 也 就 意味 着 "…。 
传 给 函数 或 方法 的 是 这 个 变量 的 一 个 副本 ， 除 非 使 用 了 引用 符号 “ 们 ”来 声明 是 要 做 一 个 引用 ， 而 
不 是 一 个 复制 。 在 PHP 5.0 中 ， 对 象 总 是 以 引用 的 形式 存在 的 ， 对 象 中 的 赋值 操作 同样 也 都 是 一 个 


引用 操作 。 
下 面 创建 一 个 案例 ， 演 示 一 下 在 PHP 中 对 象 之 间 的 引用 。 该 案例 的 代码 如 下 所 示 ; 
案例 6-9 第 
<?php 总 


class foo { 
private $x; 
function setX(Sx) { 
Sthis->x = $x; 
} 
function getX() { 


Q 
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return S$this->x; 

} 
1 
$01 = new foo; 
Sol->setX(4) 7 
$02 = $ol; 
S$Sol->setX(5) 7 
if($ol->getX() == $02->getx()) print (" 这 两 个 的 值 相等 ") ; 
2 


将 上 述 代码 保存 ， 文 件 名 为 yiny.php， 并 保存 到 指定 的 位 置 。 打 开 正 浏览 器 ， 在 地 址 栏 中 输入 
http://localhost:8080/class/yiny.php， 单 击 【 转 到 】 按 钮 ， 会 
显示 如 图 6-9 所 示 的 窗口 。 

在 上 述 代码 中 创建 了 一 个 类 foo， 在 类 中 创建 了 一 个 
变量 x， 并 为 该 变量 创建 了 相应 设置 和 获取 值 的 getXO 和 
setX0 方 法 。 在 下 面 使 用 语句 “$ol = new foo0” 创 建 了 实 
例 化 对 象 ol， 并 调用 方法 setX0 设 置 变 量 x 的 值 为 4。 语 
句 “$o2=$ol” 表 示 把 ol 的 地 址 (引用) 赋 给 o2， 这 时 o2 和 ol 同时 指向 一 个 地 址 ， 也 就 是 说 ， 对 
ol 所 做 的 任何 操作 都 会 在 引用 内 存 块 中 显示 出 来 , 02 都 会 得 到 相应 的 改变 信息 。 所 以 在 下 面 设置 变 
量 x 的 值 为 5 的 时 候 ， 该 内 存 块 保存 的 x 值 已 经 发 生 了 变化 ， 这 时 对 象 ol 和 对 象 02 的 获取 变量 x 
的 值 是 相同 的 。 

当 使 用 对 象 引 用 时 ， 如 上 案例 ，ol 的 地 址 会 把 o2 的 地 址 覆盖 ，o2 的 地 址 就 消失 ， 以 后 再 也 找 
不 到 了 。 克 隆 一 个 对 象 时 ， 克 隆 产 生 的 对 象 和 被 克隆 的 对 象 二 者 都 会 存在 ， 在 以 后 的 操作 过 程 中 ， 
二 者 之 间 的 关系 在 克隆 完成 后 ， 就 不 复 存 在 了 。 每 个 对 象 都 可 以 独立 完成 自己 的 操作 。 


6-9 对象 引用 


6.8 ”对象 的 比较 


在 PHP 5.0 中 ， 对 象 的 比较 比 在 PHP 4.0 中 更 复杂 、 更 协调 ， 当 使 用 比较 操作 符 G 疏 时 ， 对 象 


变量 将 以 一 种 简单 方式 被 比较 。 也 就 是 说 ， 如 果 它 们 具有 相同 
的 ， 并 且 是 同一 个 类 的 实例 。 另 一 方面 ， 当 使 用 恒等式 操作 符 
用 同一 个 类 的 同一 个 实例 时 是 相同 的 。 


{ public $flag; 


1 
class OtherFlag 
{ public $flag; 


} 

$0 = new Flag(); 

$p = new Flag(); 

$q = $0; 

Sr = new OtherFlag(); 

echo "同一 个 类 的 两 个 不 同 的 实例 化 对 象 <br>"; 
compareObjects ($0, $p); 

echo "<br> 两 个 相同 的 引用 指向 同一 个 类 <br>"; 
compareObjects ($0, $q); 

echo "<br> 两 个 不 同类 的 实例 化 对 象 <br>"; 
compareObjects (So， $r); 

人 人 
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属性 和 值 ， 则 两 个 对 象 的 实例 是 相等 
== 避 时 ， 对 象 变量 当 且 仅 当 它们 引 


下 面 创建 一 个 案例 ， 演 示 一 下 对 象 之 间 的 比较 。 该 案例 的 代码 如 下 所 示 : 


卫 案例 6-10 

强 <?php 

Es function bool2str ($bool) 

台 { if ($bool===false) { return 'FALSE'; } 

Ie else { return 'TRUE'; } 

并 } 

全 

学 function compareObjects (&$01, &$02) 

习 { echo 'ol ==02:'. bool2str($0l] == $02) . "<br>"; 

训 echo "ol !=02 :bool2atr(sol Ns S02 <DE>P7 
echo 'ol ===02: ' . bool2str($0l === $02) . "<br>"; 
echo 'ol !== 02 : ' . bool2str($ol !== $02) . "<br>"; 

} 
class Flag 


function Flag($flag = true) { S$this->flag = $flag; } 


function OtherFlag($flag = true) { Sthis->flag = $flag; } 


将 上 述 代码 保存 ， 文 件 名 为 bijiaophp， 并 保存 到 指定 位 置 。 打 开 正 浏览 器 ， 在 地 址 栏 中 输入 
http://localhost:8080/class/bijiao.php， 单 击 【 转 到 】 按 钮 ， 会 显示 如 图 6-10 所 示 的 窗口 。 


到 http://localhost:8080/ec 


两 个 不 同类 的 实例 化 对 象 
02 : FALSE 


图 6-10 对 象 比较 


在 上 述 代 码 中 创建 了 两 个 函数 用 来 比较 两 个 数值 是 否 相 等 ，bool2str0 函 数 中 的 参数 是 按 值 传递 
的 , compareObjects0 函 数 中 的 参数 是 按 地 址 传递 的 , 即 引 用 。 创建 两 个 类 , 分 别 为 Flag 和 OtherFlag， 
在 下 面 创建 类 Flag 的 两 个 实例 化 对 象 o。 和 p， 此 时 这 两 个 对 象 指向 不 同 的 内 存 地 址 ， 但 二 者 具有 的 
方法 和 属性 相同 。 语句“$q =$o” 表 示 两 个 对 象 指向 了 同一 块 内 存 空间 。r 是 第 二 个 类 创建 的 实例 化 
对 象 。 在 上 述 比较 中 ， 应 切记 “==” 和 “===” 的 用 法 。 


′ 学 习 目标 |oujectve 


在 应 用 程序 开发 、 测 试 及 运行 的 各 个 阶段 都 有 可 能 出 现 错误 ， 这 些 错误 可 能 是 程序 员 引起 的 ， 
也 有 些 是 开发 过 程 中 的 某 些 失误 或 其 他 原因 导致 的 。 但 不 管 怎样 ， 应 用 程序 都 必须 能 够 以 妥善 的 方 
式 处 理 这 些 预料 之 外 的 错误 ， 并 做 出 相应 的 反应 ， 以 使 数据 安全 、 完 整 ， 并 保证 程序 和 系统 的 稳定 
运行 。 另 外 ， 应 用 程序 应 当 能 为 用 户 提供 必要 的 反馈 信息 ， 以 使 用 户 有 针对 性 地 解决 出 现 的 错误 。 


oe 
-= 内 容 摘要 |avstrsct 


> 掌握 配置 指令 
> 理解 错误 日 志 
> 掌握 异常 处 理 


> 理解 为 什么 要 进行 异常 处 理 


> 掌握 异常 处 理 的 实现 


7.1 配置 指令 


在 PHP 中 , 许多 配置 指令 确定 了 错误 的 报告 行为 。 由 于 这 样 的 配置 指令 较 多 ， 本 节 只 介绍 一 些 
比较 常见 的 指令 ， 如 表 7-1 所 示 。 


指 令 


erTOT Teporting(String) 


display_errors(On. Of 


表 7-1 PHP 的 常见 指令 


说 阴 

该 指令 确定 报告 的 敏感 级 别 , 它 的 作用 域 为 PHP_INI ALL; 默认 值 为 E_ALL $ 一 
E_STRICT。 在 PHP 中 共有 13 个 不 同 的 级 别 ， 如 表 7-2 所 示 。 这 些 级 别 的 任何 
组 合 都 是 有 效 的 ， 并 且 每 个 级 别 都 包括 位 于 其 下 面 的 所 有 级 别 

启用 该 指令 时 ， 将 显示 满足 error_reporting 下 定义 规则 的 有 错误 ， 所 以 应 该 在 进 
行 测试 期 间 启 用 此 指令 , 在 网 站 投入 使 用 时 要 将 其 禁用 。 因为 显示 这 些 消息 不 仅 
可 能 会 让 终端 用 户 更 迷惑 , 还 有 可 能 会 过 多 地 泄露 有 关 应 用 及 服务 器 的 信息 。 该 
指令 的 作用 域 为 PHP_INI ALL， 默认 值 为 On 


display_startup_errors(On. Of 


启用 该 指令 将 显示 PHP 引擎 初始 化 时 遇 到 的 所 有 错误 。 与 display_errors 类 似 ， 
应 该 在 测试 时 启用 该 指令 ， 在 网 站 投入 使 用 时 将 其 禁用 。 该 指令 的 作用 域 为 
PHP INI ALL; 默认 值 为 Off 


log_errors(On. Off) 


该 指令 用 于 记录 应 用 程序 和 PHP 引擎 在 运行 测试 过 程 中 产生 的 错误 ， 通 过 错误 
记录 可 以 为 某 些 特定 问题 的 解决 提供 有 价值 的 信息 。 因 此 ， 应 当 始 终 启 用 
log _ermors。 这 些 日 志 语句 记录 的 位 置 取决 于 error log 指令 。 该 指令 的 作用 域 为 
PHP_INI ALL; 默认 值 为 Of 


说 了 明 
在 PHP 中 , 错误 可 以 发 送 给 系统 syslog, 或 者 送 往 由 管理 员 通过 error log 指令 
指定 的 文件 。 如 果 该 指令 设置 为 syslog， 则 在 Linux 上 错误 语句 将 送 往 syslog， 
而 在 Windows 上 错误 语句 将 送 往事 件 日 志 。 其 中 syslog 是 基于 UNIX 的 日 志 工 
有 具 ,提供 了 一 个 API 来 记录 与 系统 和 应 用 程序 执行 有 关 的 消息 。Windows 事件 
日 志 实际 上 与 UNIX 的 syslog 相同 .这 些 日 志 通常 可 以 通过 事件 查看 器 来 查看 。 
该 指令 的 作用 域 为 PHP_INI ALL; 默认 值 为 NULL 
该 指令 用 于 设置 每 个 日 志 项 的 最 大 长 度 ， 以 B 为 单位 。 默 认 值 为 1024B。 将 该 
指令 设置 为 0 表示 不 指定 最 大 长 度 。 该 指令 的 作用 域 为 PHP_INI_ALL; 默认 值 
为 1024 
启用 该 指令 将 使 PHP 忽略 在 同一 文件 中 同一 行 上 发 生 的 重复 错误 消息 。 该 指令 
的 作用 域 为 PHP_INI ALL; 默认 值 为 Of 
启用 该 指令 将 使 PHP 忽略 在 同一 文件 中 或 同一 文件 中 不 同行 上 发 生 的 重复 错 
误 消 息 。 该 指令 的 作用 域 为 PHP_INI ALL; 默认 值 为 Of 
启用 该 指令 将 使 PHP 在 变量 php_errormsg 中 存储 最 近 发 生 的 错误 消息 ,一 旦 注 
册 ， 就 可 以 随心 所 欲 地 使 用 此 变量 数据 。 例 如 输出 、 存 储 到 数据 库 或 其 他 可 以 
对 变量 做 的 事情 。 该 指令 的 作用 域 为 PHP INI ALL; 默认 值 为 Of 


表 7-2 ”PHP 的 错误 报告 级 别 


错误 报告 级 别 


指 令 


error log(string) 


log errors max len(integer) 


ignore repeated error(On, Off) 
ignore repeated_source(On, Of 


track_errors(On, Of 


E ALL 
E ERROR 
E_WARNING 
E_PARSE 
E NOTICE 
E_STRICT 


E_CORE ERROR PHP 开始 启动 时 发 生 的 致命 
错误 


这 里 需要 注意 的 是 ， 表 7-2 中 的 每 个 级 别 都 包括 位 于 其 下 的 所 有 级 别 。 例 如 ，E_ERROR 会 报 
告 表 中 在 它 之 下 的 所 有 11 个 级 别 的 消息 。 其 中 ，E_STRICT 是 PHP 5.0 的 新 内 容 ， 它 建议 基于 核心 
开发 人 员 对 编码 方法 的 决定 来 修改 代码 ， 从 而 确保 不 同 PHP 版 本 之 间 的 可 移植 性 。 如果 使 用 了 已 经 
废弃 的 函数 或 语法 、 不 正确 地 使 用 了 引用 ， 对 类 字段 使 用 var 而 不 是 作用 域 ， 或 者 引入 了 异样 的 风 
格 ，E_STRICT 都 会 提醒 用 户 注意 。 

另外 需要 注意 的 是 ， 逻 辑 操作 符 Not 用 一 字符 来 表示 。 这 种 含义 只 在 这 个 指令 中 有 效 ， 而 对 于 
PHP 中 的 其 他 部 分 ， 都 使 用 惊叹 号 〈!) 表示 Not。 

error reporting (E_ALL) 这 样 的 设置 可 以 用 于 开发 阶段 ， 因 为 ， 在 进行 开发 时 希望 报告 所 有 的 
错误 。 

error_reporting (E_ ERROR | E COMPILE ERROR) 这 样 的 设置 可 以 只 报告 致命 的 运行 时 错误 及 
致命 的 编译 时 错误 。 

error reporting (E_ALL | ~ (E_COMPILE ERROR | E COMPILE WARNING) ) 这 样 的 设置 可 
以 只 报告 除 编译 错误 之 外 的 所 有 错误 。 该 种 设置 在 实际 应 用 中 非常 常见 ， 因 为 适当 的 错误 报告 有 利 
于 程序 开发 过 程 中 的 问题 解决 。 
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7.2 错误 日 志 


在 PHP 中 , 错误 日 志 可 以 存放 在 syslog 中 ,也 可 以 存放 在 某 个 单独 的 文本 文件 中 ， 具体 使 用 哪 
一 种 方式 取决 于 用 户 所 处 的 环境 。 如 果 网 站 在 共享 的 服务 器 上 运行 ， 那 么 使 用 单独 的 文本 文件 或 数 
据 库 表 可 能 是 唯一 的 选择 ， 如 果 用 户 可 以 控制 服务 器 ， 使 用 syslog 则 较为 理想 ， 因 为 用 户 可 以 利用 
syslog 的 解析 工具 来 查看 和 分 析 日 志 。 所 以 用 户 在 进行 选择 时 要 根据 自己 的 实际 情况 来 进行 。 

如 果 要 使 用 单独 的 文本 文件 来 记录 错误 日 志 , 那么 Web 服务 器 进行 程序 所 有 者 必须 有 足够 的 权 
限 来 写 这 个 文件 。 为 了 减少 遭 到 攻击 的 可 能 性 ， 应 该 将 这 个 文件 存放 在 文档 根 目录 之 外 。 如 下 所 示 
内 容 类 似 于 写 入 单独 文本 文件 中 的 消息 : 

[Thu Jun 14 15:35:19 2007] [error] 


[client 127.0.0.1] 
script 'C:/Apache2.2/htdocs/test.php' not found or unable to stat 


1. define_syslog_variables() 

它 的 使 用 格式 如 下 所 示 : 

void define syslog variables (void); 

该 函数 初始 化 一 些 常 量 ， 这 些 常 量 是 使 用 openlog0、closelog0 和 syslog0 函 数 时 所 必需 的 ， 使 
用 它们 之 前 必须 先 执行 define_syslog variables0 函 数 。 

2. openlog() 

它 的 使 用 格式 如 下 所 示 : 


int openlog(string ident,int option,int facility); 


该 函数 打开 一 个 与 所 在 平台 上 系统 日 志 器 的 连接 ， 通 过 指定 几 个 将 在 日 志 上 下 文 使 用 的 参数 ， 
为 向 系统 日 志 插入 消息 做 好 准备 。 它 的 各 参数 介绍 如 下 所 示 。 
(1) ident: 它 表示 增加 到 某 一 项 开始 处 的 消息 标识 符 。 通 常 这 个 值 设 置 为 程序 名 。 因 此 ， 用 户 
可 能 会 把 PHP 有关 的 消息 标识 为 “PHP” 或 “PHP5”。 
(2) option: 它 用 于 确定 生成 消息 时 使 用 哪些 日 志 选 项 。 如 表 7-3 所 示 列 出 了 可 用 的 日 志 选 项 。 
表 7-3 日 志 选 项 


LOG CONS 如 果 写 入 syslog 时 发 生 错 误 ， 则 将 发 送 到 系统 控制 台 


LOG NDELAY 立即 打开 与 syslog 的 连接 

LOG ODELAY 不 打开 连接 ， 直 到 提交 了 第 一 条 消息 为 止 。 这 是 默认 值 
LOG PERROR 记录 的 消息 同时 输出 到 syslog 和 标准 错误 (standard error) 
LOG PD 每 个 消息 都 带 有 进程 ID (PID) 


如 果 同 时 需要 上 述 多 个 选项 ， 可 以 将 各 选项 间 用 竖 线 “|” 进 行 分 隔 。 例 如 ， 可 以 指定 如 下 所 示 
的 三 个 选项 : LOG_CONS | LOG PERROR |LOG PID。 
(3) facility: 它 有 助 于 确定 记录 消息 日 志 的 程序 属于 哪 一 类 。 它 的 值 可 能 为 LOG KERN、 


LOG USER\ LOG _ MAIL、LOG _ DAEMON\、LOG AUTH\、LOG SYSLOG、LOG LPR、LOG NEWS、 


LOG UUCP、 


LOG CRON 或 LOG AUTHPRIV。 其 中 ,指定 LOG_USER 会 使 消息 发 送 到 messages 


文件 ， 只 有 当 
将 使 后 继 的 消 


3. closel 


PHP 用 作 一 个 命令 行 解 释 器 时 才 会 将 这 个 参数 设置 为 LOG_USER。 指 定 LOG_CRON 
息 发 送 到 cron 日 志 。 
log() 


它 的 使 


格式 如 下 所 示 : 


int closelog(void); 


该 函数 用 来 关闭 已 打开 的 系统 纪录 。 它 无 传 入 参数 ， 也 不 是 必需 的 函数 ， 因 为 PHP 程序 在 执行 
完成 后 会 自动 关闭 开启 的 资源 。 

4. syslog() 

它 的 使 用 格式 如 下 所 示 : 


int syslogl(int priority, string message ); 


程度 。 它 的 可 


message 字符 串 写 到 系统 记录 中 , 参数 priority 用 于 指定 syslog 优先 级 , 这 里 表示 严重 
取 值 及 其 说 明 如 表 7-4 所 示 。 该 函数 呼叫 UNIX 作业 系统 的 syslog0 函 数 ， 在 Windows 


NT 上 ， 使 用 事件 监视 器 模拟 本 功能 。 


表 7-4 ”参数 priority 选 项 


选 项 说 了 明 
LOG EMERG 严重 的 系统 问题 ， 可 能 预示 着 崩溃 
LOG ALERT 必须 解决 的 情况 ， 可 能 危害 系统 完整 性 
LOG CRIT 紧急 错误 ， 可 能 导致 服务 不 可 用 ， 但 不 一 定 会 使 系统 陷入 危险 
LOG ERR 一 般 错 误 
LOG WARNING 一 般 警 告 
LOG NOTICE 正常 但 值得 注意 的 情况 
LOG INFO 一 般 信息 
LOG DEBUG 一 般 只 与 调试 应 用 程序 有 关 的 信息 


参数 message 指定 要 记录 的 文本 消息 。 如 果 希 望 记 录 由 PHP 引擎 提供 的 错误 消息 ， 就 可 以 在 


message 中 包 
代 蔡 。 
5. 错误 


括 字 符 串 %m。 此 字符 串 将 被 PHP 引擎 在 运行 时 提供 的 错误 消息 字符 串 (strerror) 所 


日 志 示 例 


介绍 了 错误 日 志 相关 的 函数 及 其 说 明 ， 下 面 来 看 一 个 示例 : 


<?php 


define syslog variables(); 


openlog 


("php pro", LOG PID | LOG CONS, LOG USER); 


syslog (LOG_INFO，"php_pro 错误 日 志 示例 ! "); 
closelog(); 


2 


这 段 代码 将 在 message syslog 文件 中 生成 类 似 于 下 面 的 一 条 日 志 : 


Jun 20 


11:43:01 2007 php_pro[30711]: php_pro 错误 日 志 示例 ! 
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7.3 “异常 处 理 


在 许多 语言 中 ， 如 C++、C#、Python 和 Java 等 ， 异 常 处 理 一 直 都 是 错误 管理 方面 的 中 流 碟 柱 ， 
它 为 建立 标准 化 的 错误 报告 逻辑 提供 了 一 种 绝 佳 的 方法 。 由 于 在 结构 化 编程 语言 中 设计 错误 处 理 策 
略 时 ， 不 仅 非 常 容易 出 错 ， 而 且 很 难保 持 一 致 ， 所 以 在 PHP 5.0 的 程序 设计 中 添加 了 类 似 于 其 他 语 
言 的 异常 处 理 模块 。 


7.3.1 异常 处 理 原因 


在 程序 开发 、 测 试 、 运 行 及 维护 的 各 个 阶段 随时 都 有 可 能 出 现 不 可 预见 的 事件 打 断 正常 的 事件 
链 ， 这 些 不 可 预见 的 事件 称 为 异常 。 为 了 能 正确 、 及 时 地 处 理 出 现 的 异常 ， 许 多 编程 语言 通过 抛 出 
异常 ， 也 就 是 放 一 个 代码 来 处 理 错误 ， 然 后 对 异常 妥善 地 做 出 响应 。 接 下 来 ， 异 常 处 理 器 捕获 异常 
并 进行 相应 的 处 理 。 

关于 异常 处 理 的 过 程 ， 大 多 数 编程 语言 将 其 抽象 为 4 个 步骤 ， 具体 如 下 所 示 : 

(1) 应 用 程序 尝试 进行 处 理 。 

(2) 如 果 尝 试 失败 ， 则 将 该 异常 抛 出 。 

(3) 指定 的 异常 处 理 器 捕获 异常 ， 并 进行 相应 的 处 理 。 

(4) 最 后 消除 在 尝试 处 理 异 常 期 间 占 用 的 资源 。 

通过 try/catch 的 方式 来 处 理 异常 是 许多 编程 语言 所 采用 的 语法 ， 下 面 所 示 的 伪 代 码 只 是 用 于 说 
明 try/catch 的 使 用 语法 ， 它 并 不 能 真正 运行 。 


<?php 
//try 代码 块 开始 
try { 
完成 某 些 操作 (处 理 语句 ) 
if (wrong) ” // 如 果 出 现 错误 
throw exception (" 抛 出 异常 ") ; 
// 捕 获 抛 出 的 异常 
} catch (Exception S$e) { 
// 输 出 捕获 的 异常 信息 
echo "捕获 的 异常 :"， Se->getMessage () 
} 
//try 代码 块 结束 


人 


往往 出 现 的 异常 并 非 只 有 一 个 ， 所 以 通常 可 以 创建 多 个 异常 处 理 器 来 解决 这 些 问 题 。 到 目前 为 
止 , 在 PHP 中 只 提供 一 个 简单 的 处 理 器 exception。 如 果 需 要 ， 可 以 扩展 这 个 处 理 器 ， 所 以 用 户 可 以 
使 用 预定 义 处 理 器 ， 或 扩展 某 个 预定 义 处 理 器 来 创建 满足 自己 需要 的 定制 处 理 器 。 下 面 所 示 的 伪 代 
码 是 基于 上 述 伪 代 码 的 示例 ， 用 于 说 明 多 异常 处 理 器 块 的 使 用 ， 它 也 不 能 真正 运行 。 


<?php 
//try 代码 块 开 始 


try { 
完成 某 些 操作 (处 理 语句 ) 
if (wrong1) // 如 果 出 现 错误 
throw IOexception (" 抛 出 异常 1") 7 
if (wrong2) // 如 果 出 现 错误 
throw Numberexception(" 抛 出 异常 2"); 
// 捕 获 抛 出 的 异常 1 
} catch (Exception Sel) { 
// 输 出 捕获 的 异常 信息 
echo "捕获 的 异常 :"， Sel->getMessage ()7 
} // 捕 获 抛 出 的 异常 2 
catch (Exception $e2) { 
// 输 出 捕获 的 异常 信息 
echo "捕获 的 异常 :"， $e2->getMessage () 7 


1 

//try 代码 块 结束 

?> 

对 于 初学 者 来 说 ， 异 常 处 理 对 于 标识 和 报告 应 用 程序 错误 提供 了 通用 策略 ， 另 外 ， 对 于 指定 程 
序 在 遇 到 错误 时 如 何 进行 处 理 也 有 一 般 性 的 策略 ， 通 过 使 用 这 些 策略 ， 异 常 处 理 要 优 于 错误 管理 过 
程 。 此 外 ， 异 常 处 理 的 语法 促进 了 处 理 器 与 一 般 应 用 程序 逻辑 分 离 ， 因 而 可 以 得 到 更 有 组 织 、 更 具 
可 读 性 的 代码 。 


7.3.2 ”实现 异常 处 理 


在 PHP 代码 中 产生 的 异常 可 被 throw 语句 抛 出 并 被 catch 语句 捕获 。 需 要 进行 异常 处 理 的 代码 
都 必须 放 入 try 代码 块 内 ， 以 便 捕获 可 能 存在 的 异常 。 每 一 个 try 至 少 要 有 一 个 与 之 对 应 的 catch。 
使 用 多 个 catch 可 以 捕获 不 同 的 类 产生 的 异常 。 当 try 代码 块 不 再 抛 出 异常 或 者 找 不 到 catch 能 匹配 
所 抛 出 的 异常 时 ，PHP 代码 就 会 跳 转 到 最 后 一 个 catch 的 后 面 继续 执行 。 当 然 ，PHP 允许 在 catch 
代码 块 内 再 次 抛 出 〈throw) 异常 。 

当 一 个 异常 被 抛 出 时 ， 其 后 ( 指 抛 出 异常 时 所 在 的 代码 块 ) 的 代码 将 不 会 继续 执行 ， 而 PHP 就 
会 尝试 查找 第 一 个 能 与 之 匹配 的 catch。 如 果 一 个 异常 没有 被 捕获 ， 而 且 又 没有 使 用 set_exception_ 
handler0 作 相应 处 理 的 话 ， 那 么 PHP 将 会 产生 一 个 严重 的 错误 ， 并 且 输 出 Uncaught Exception. (未 
捕获 异常 ) 的 提示 信息 。 

1. PHP 的 基本 异常 类 

PHP 的 基本 异常 类 提供 了 一 个 不 带 参 数 的 默认 构造 函数 ， 一 个 带 有 两 个 可 选 参数 的 重 载 构 造 
函数 ， 并 且 异 常 类 具有 6 个 方法 ， 以 获得 当前 异常 返回 的 相关 信息 。 本 节 将 分 别针 对 这 些 内 容 进行 
介绍 。 

(1) 默认 构造 函数 

默认 构造 函数 不 带 参数 ， 用 户 可 以 用 如 下 方式 使 用 异常 类 : 


throw new Exception(); 


以 上 抛 出 的 异常 在 实例 化 后 就 可 以 使 用 异常 类 所 具有 的 6 个 方法 中 的 4 个 ， 另 外 两 个 只 有 在 使 
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重 载 构造 函数 实例 化 异常 类 时 才能 使 用 。 

(2) 重 载 构造 函数 
重 载 构造 函数 可 以 有 两 个 参数 ， 由 此 能 提供 默认 构造 函数 所 没有 的 其 他 功能 。 它 的 使 用 格式 如 
下 所 示 : 


throw new Exception (message,error code); 


其 中 ， 参 数 message 用 于 向 用 户 提供 关于 出 现 异 常 的 信息 ， 此 信息 可 以 通过 异常 类 的 方法 
getMessage() 获 得 。 参 数 error code 用 于 保存 错误 标识 符 ， 它 可 以 映射 到 某 个 标识 符 - 消 息 表 。 而 错误 
标识 符 通常 用 于 国际 化 和 本 地 化 。 在 这 里 错误 标识 符 可 以 通过 异常 类 的 方法 getCode0 得 到 。 如 下 所 
示 是 重 载 构造 函数 的 使 用 示例 : 

throw new Exception("1 is an invalid parameter', 5); 


throw new Exception('l1 is an invalid parameter'); 
throw new Exception(, 5); 


(3) 方法 
在 PHP 中 为 了 获得 所 抛 出 异常 的 相关 信息 ， 为 异常 类 提供 了 6 个 方法 。 这 6 个 方法 及 其 说 明 如 
表 7-5 所 示 。 


册 


表 7-5 异常 类 的 方法 


法 说 了 明 
getMessage() 返回 传递 给 构造 函数 的 消息 
getCode0) 返回 传递 给 构造 函数 的 错误 代码 
getLine0) 返回 抛 出 异常 的 行 号 
getFile0) 返回 抛 出 异常 的 文件 名 
getTrace() 返回 一 个 数组 ， 其 中 包括 出 现 错误 的 上 下 文 的 有 关 信 息 。 通 常情 况 下 ， 该 数组 包括 文件 名 、 


行 号 、 函 数 名 和 函数 参数 
getTraceAsString() | 返回 与 getTrace0 方 法 完全 相同 的 信息 ， 只 是 返回 的 信息 是 一 个 字符 串 而 不 是 数组 


(4) 简单 的 异常 处 理 
前 面 已 经 对 PHP 基本 异常 类 及 其 相应 方法 进行 了 介绍 ， 下 面 来 看 一 个 简单 的 案例 。 该 案例 是 一 
个 简单 的 产生 异常 ， 并 对 其 进行 处 理 的 案例 ， 具 体 代码 如 下 所 示 : 


案例 7-1 
<?php 
echo "try 代码 块 开始 <br>"; 
try { 
Snuml = 100; 
Snum2 = 0; 
if ($num2==0) 


$error = "除数 不 能 为 01"; 
throw new Exception($error); 
' 

else 


‘ 


$result = Snuml/Snum27 

echo "$result"; 

# 

// 产生 异常 后 ， 下 面 的 try 代码 块 内 的 代码 将 不 会 被 执行 “ 
echo "<br> 输 出 此 信息 表示 前 面 未 产生 异常 ! "7 2 


} catch (Exception $e) { 

echo "捕获 的 异常 :"， $e->getMessage(); 
1 
echo "<br>try 代码 块 结束 "; 


// 继续 执行 
echo "<br> 这 里 代码 在 try 代码 块 外 ."; 
2 第 
7 
将 上 述 代码 存储 在 7-1.php 文件 中 ， 保 存 到 C:\Apache2.2\htdocs\7 中 ， 然 后 打开 正 浏览 器 ， 在 章 
地 址 栏 中 输入 http://localhost/7/7-1.php， 运 行 结果 如 下 所 示 : 错 
案例 7-1 和 
try 代码 块 开 始 异 
捕获 的 异常 :除数 不 能 为 0! 常 
try 代码 块 结束 处 
这 里 代码 在 try 代码 块 外 . 理 


2. 内 置 异常 处 理 类 的 结构 


处 理 类 中 ， 哪 些 属性 和 方法 在 子 类 中 是 可 以 访问 和 继承 的 ， 所 以 以 下 这 段 代 码 只 为 说 明 内 置 异 常 处 
理 类 的 结构 ， 它 并 不 是 一 段 有 实际 意义 的 可 用 代码 。 


<?php 

class Exception 

{ 
protected $message = 'Unknown exception'; 
protected $code = 0; 


// 异常 信息 
// 用 户 自 定义 异常 代码 


protected $file; 
protected $line; 


// 发 生 异 常 的 文件 名 
// 发 生 异 常 的 代码 行 号 


function __construct($message = null, $code = 0); 
final function getMessage(); // 返回 异常 信息 
final function getcode () 7 // 返回 异常 代码 


final function getrFile(); 

final function getLine(); 

final function getTrace(); 

final function getTraceAsstring(); 


/* 可 重 载 的 方法 */ 


function __tostring(); 


// 返回 发 生 异 常 的 文件 名 

// 返回 发 生 异 常 的 代码 行 号 

// backtrace() 数组 

// 已 格式 化 成 字符 串 的 getTrace () 信息 


// 可 输出 的 字符 串 
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3. 使 用 自 定义 类 处 理 异 常 

如 果 使 用 自 定 义 的 类 来 扩展 内 置 异常 处 理 类 ， 并 且 要 重新 定义 构造 函数 ， 建 议 同 时 调用 
parent:: _construct0 来 检查 所 有 的 变量 是 否 已 被 赋值 。 当 对 象 要 输出 字符 串 时 , 可 以 重 载 _ toString0 
并 自 定义 输出 的 样式 。 下 面 通 过 扩展 PHP 内 置 异常 处 理 类 来 实现 PHP 中 的 异常 处 理 。 具 体 代 码 如 


下 所 示 : 


Case self::THROW DEFAULT: 
// 抛 出 默认 的 异常 
throw new Exception('2 isnt allowed as a parameter', 6); 
break; 


default: 
// 在 没有 异常 的 情况 下 创建 一 个 对 象 


$this->var = $avalue; 


break; 
} 
} 
} 第 
7 
音 
// 测试 异常 处 理 1 = 
try { 错 
$0 = new TestException (TestException: :THROW CUSTOM); 误 
} catch (MyException $e) { // 捕获 异常 和 
echo "Caught my exception\n", $e; 异 
$e->customFunction(); 处 
} catch (Exception $e) { // 被 忽略 理 
echo "Caught Default Exception\n", $e; 
上 
// 执行 后 续 代码 


var_ dump ($0); 
Ceo NN 


// 测试 异常 处 理 2 
try { 
$0 = new TestException (TestException::THROW DEFAULT); 
} catch (MyException $e) { // 不 能 匹配 异常 的 种 类 ， 被 忽略 
echo "Caught my exception\n", $e; 
$e->customFunction(); 
} catch (Exception $e) { // 捕获 异常 
echo "Caught Default Exception\n", $e; 
出 


// 执行 后 续 代 码 
var_ dump ($0); 
echo "\n™s 


// 测试 异常 处 理 3 
try { 

$0 = new TestException(TestException::THROW CUSTOM); 
} catch (Exception $e) { // 捕获 异常 


echo "Default Exception caught\n", $e; 


| 
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将 上 述 代码 存储 在 7-2.php 文件 中 ， 保 存 到 C:\Apache2.2\htdocs\7 中 ， 然 后 打开 正 浏览 器 ， 在 
地 址 栏 中 输入 http://localhost/7/7-2.php， 运 行 结 果 如 下 所 示 : 


第 8 章 字符 串 和 正则 表达 式 


二 上 学 习 目 标 |objectve 


在 每 一 种 语言 设计 的 程序 中 ， 不 可 避免 地 都 要 处 理 大 量 的 字符 串 信 息 。 对 于 字符 串 的 功能 处 理 
显示 了 一 个 语言 的 成 功 与 否 。 在 PHP 中 ,提供 了 100 多 个 函数 对 字符 串 进行 处 理 ， 按 处 理 方式 的 不 
同 分 为 两 种 类 型 ， 一 种 是 基于 正则 表达 式 的 函数 ， 一 种 是 普通 的 字符 串 函 数 。 

本 章 将 从 正则 表达 式 开始 ， 介 绍 正则 表达 式 的 创建 和 使 用 ， 以 及 常用 的 一 般 字符 串 函数 。 


SS 
2 内 容 摘要 |avstrsct 


了 解 PHP 中 支持 使 用 大 括号 语法 

掌握 PHP 中 两 种 不 同 风格 的 正则 表达 式 语 法 

掌握 PHP 中 基于 Perl 的 正则 表达 式 函 数 

掌握 PHP 中 基于 POSIX 的 正则 表达 式 函 数 

掌握 PHP 中 比较 字符 串 大 小 函数 

掌握 PHP 中 字符 串 的 大 小 写 转换 函数 

掌握 PHP 中 普通 文本 信息 和 HTML 信息 转换 函数 
掌握 PHP 中 正则 表达 式 函 数 


8.1 复杂 (大 括号 ) 偏 移 语法 


PHP 是 一 种 弱 类 型 语言 ， 在 处 理 字 符 串 时 ， 可 以 把 字符 串 作为 数组 来 处 理 。 因 为 每 个 字符 串 都 
是 由 连续 的 字符 组 成 的 集合 ， 也 可 以 看 作 一 个 连续 的 实体 。 下 面 的 代码 反映 了 字符 串 既 可 以 作为 字 
符 串 处 理 也 可 以 作为 数组 处 理 。 


<?php 
$str="hello"; 
echo $str; 
echo "<br>"7 
echo $str[0]; 
echo $str[1]; 
echo S$str[2]; 
echo $str[3]; 
echo S$str[4]; 

有 2 区 


在 上 述 代 码 中 可 以 看 到 ， 使 用 数组 的 输出 结果 和 使 用 字符 串 的 输出 结果 是 一 样 的 。 在 这 里 使 用 
中 括号 来 表示 字符 串 的 数组 形式 的 输出 ， 可 能 会 和 数组 产生 二 义 性 ， 因 为 数组 也 是 使 用 中 括号 的 。 
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在 这 里 ， 可 以 使 用 大 括号 来 代替 中 括号 。 其 形式 如 下 所 示 : 

<?php 
$str="hello™; 
echo $str; 
echo "<br>"; 
echo $str{0}; 
echo $str{1}; 
echo $str{2}; 
echo $str{3}; 
echo $str{4}; 

?> 


使 用 大 括号 代替 中 括号 ， 完 全 可 以 消除 语法 上 的 二 义 性 。 推 荐 使 用 这 种 形式 。 


8.2 正则 表达 式 


正则 表达 式 的 使 用 使 程序 员 在 校 验 字符 串 时 ， 节 省 了 大 量 的 代码 ， 并 以 更 高 的 效率 执行 PHP 页 
面 程序 。 本 节 将 详细 地 介绍 PHP 中 正则 表达 式 的 使 用 。 


8.2.1 简介 


在 创建 PHP 页 面 的 过 程 中 , 不 可 避免 地 要 对 某 些 字符 串 的 格式 进行 校 验 , 如 电子 邮件 的 格式 需 
要 包含 两 个 特殊 字符 “@” 和 “.”， 或 者 电话 号 码 的 格式 。 对 于 这 种 情况 需要 编写 相应 的 函数 对 输 
入 的 字符 串 进行 验证 ， 符 合 条 件 的 允许 通过 ， 否 则 不 允许 通过 。 当 整个 Web 程序 完成 之 后 ， 可 能 会 
编写 大 量 的 验证 函数 来 完成 验证 字符 串 的 工作 。 

正则 表达 式 的 功能 和 这 些 函 数 相同 ， 而 且 更 好 一 些 。 正 则 表达 式 可 以 被 认为 是 一 个 极其 高 级 的 
查找 -替换 验证 工具 ,使 用 户 从 字符 串 操作 的 痛苦 中 摆脱 出 来 : 不 必 再 写 定制 的 数据 确认 例子 来 检查 
电子 邮件 地 址 或 者 电话 号 码 的 格式 是 否 正 确 ， 如 此 等 等 。 任 何 程序 中 最 普通 的 函数 之 一 就 是 数据 有 
效 性 检查 ，PHP 捆绑 了 一 些 文本 检查 函数 ， 允 许 用 户 用 正则 表达 式 匹配 一 个 字符 串 ， 确 认 是 否 存在 
空格 或 问号 等 。 

下 面 看 两 段 程序 代码 ， 第 一 段 代 码 如 下 所 示 

<?php 

function validateEmail ($email) 

1 

ShasRAtSymbol = strpos (Semai1l，"@") 

ShasDot = strpos (Semail，".")7 

if(S$hasRAtSymbol && ShasDot) 

return true; 

5 


return false; 
echo validateEmail ("mitchell@devarticles.com"); 
2 


该 段 代码 主要 实现 使 用 函数 来 对 字符 串 进 行 验证 ， 这 里 验证 的 是 是 否 符合 电子 邮件 的 格式 。 第 
二 段 代 码 如 下 所 示 : 

<?php 

function validateEmail ($email) 

1 

return ereg("^[a-zA-Z]+@[a-zA-Z]+\. [a-zA-Z]+$", S$email); 

1 

echo validateEmail ("mitchell@devarticles.com"); 

2 


该 段 代码 也 是 实现 对 电子 邮件 格式 的 验证 功能 。 从 上 面 两 段 代 码 可 以 了 解 到 ， 第 一 个 函数 比较 
容易 ， 而 且 结 构 简单 ， 但 是 如 果 用 上 面 的 下 一 个 版 本 的 Email 地 址 检查 函数 更 容易 。 上 面 展 示 的 第 
二 个 函数 只 用 了 正则 表达 式 ， 包 括 了 对 ereg0 函 数 的 一 个 调用 。ereg0 函 数 返回 True 或 者 False， 来 
声明 它 的 字符 串 参 数 是 否 和 正则 表达 式 相 匹配 。 

很 多 程序 开发 人 员 避 开 正 则 表达 式 ， 只 因为 它们 (在 一 些 情况 下 ) 比 其 他 的 文本 处 理 方法 更 慢 。 
正则 表达 式 可 能 慢 的 原因 是 因为 它们 涉及 把 字符 串 在 内 存 中 复制 和 粘贴 ， 因 为 正则 表达 式 的 每 一 个 
新 的 部 分 都 对 应 匹配 一 个 字符 串 。 但 是 ， 除 非 在 文本 中 几 百 个 行 运行 一 个 复杂 的 正则 表达 式 ， 否 则 
性 能 上 的 缺陷 都 可 以 忽略 不 计 ， 当 把 正则 表达 式 作 为 输入 数据 检查 工具 时 ， 也 很 少 出 现 这 种 情况 。 

正则 表达 式 是 由 普通 字符 (例如 字符 a 一 z) 以 及 特殊 字符 〈 称 为 元 字符 ) 组 成 的 文字 模式 。 正 
则 表达 式 作 为 一 个 模板 ， 将 某 个 字符 模式 与 所 搜索 的 字符 串 进行 匹配 。 一 个 正则 表达 式 是 一 个 特定 
的 格式 化 模式 , 可 以 用 来 找 出 一 个 字符 串 在 另 一 个 字符 串 中 的 使 用 情况 。 几 种 编程 语言 , 包括 Visual 
Basic、Perl、JavaScript 都 支持 正则 表达 式 。PHP 有 两 套 函 数 用 来 处 理 两 种 类 型 的 正则 表达 式 : Perl 
风格 兼容 模式 和 POSIX 风格 兼容 模式 。 可 以 通过 在 一 对 分 隔 符 之 间 放 入 表达 式 模 式 的 各 种 组 件 来 构 
造 一 个 正则 表达 式 ， 即 /expression/。 

POSIX 正则 表达 式 比 Perl 兼容 的 功能 弱 ， 并 且 有 时 速度 慢 , 但 是 易于 阅读 。 正 则 表达 式 有 三 种 
作用 : 匹配 、 蔡 换 和 拆 分 ， 即 用 于 从 字符 串 中 提取 信息 ; 用 新 文本 取代 匹配 的 文本 ;把 字符 串 拆 分 
成 小 块 的 数组 。PHP 为 Perl 和 POSIX 正则 表达 式 中 的 这 三 种 行为 都 提供 了 函数 。 例 如 ，ereg0O 函 数 
进行 POSIX 匹配 ， 而 preg_matchO) 函 数 进行 Perl 匹配 。 幸 运 的 是 ， 基 本 的 POSIX 和 Perl 正则 表达 
式 之 间 有 一 些 相似 之 处 。 


8.2.2 ”POSIX 正则 表达 式 语 法 


POSIX 正则 表达 式 的 结构 和 一 般 的 数学 表达 式 相似 ， 各 个 元 素 (操作 符 ) 组 合 在 一 起 ， 构 成 一 
个 更 复杂 的 表达 式 。 这 种 组 合 决定 了 不 仅 可 以 找到 或 匹配 直接 量 (字面 ) 表达 式 ， 如 某 个 单词 或 数 
字 ， 还 可 以 找到 许多 语义 不 同 但 语法 相似 的 字符 串 ， 如 文件 中 的 HTML 标记 。 

POSIX 主要 使 用 UNIX 地 区 系统 (locale system)， 该 系统 提供 了 排序 和 识别 字符 的 函数 ， 以 智 
能 地 处 理 其 他 非 英语 的 文本 。 特 别 地 ， 各 种 语言 组 成 单词 的 “字母 ”( 如 a 和 c) 不 同 ，POSIX 正则 
表达 式 考 虑 到 了 这 一 点 并 提供 相应 的 字符 类 。 

然而 ，POSIX 正则 表达 式 是 为 使 用 仅 有 原文 的 数据 而 设计 的 。 如 果 数 据 中 有 空 字 节 (x00), 那 
么 正则 表达 式 函 数 把 它 理解 为 字符 串 的 末尾 ， 并 且 匹 配 不 会 超过 该 字 节 位 置 。 要 匹配 任意 的 二 进 制 
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数据 ， 需 要 使 用 兼容 Perl 的 正则 表达 式 ， 因 为 Perl 风格 的 正则 表达 式 函数 常常 比 等 效 的 POSIX 风 
格 的 函数 快 。 
在 介绍 PHP 基于 POSIX 的 正则 表达 式 函 数 之 前 ， 首 先 介绍 POSIX 为 定位 不 同 的 字符 序列 支持 
的 三 种 语法 ， 中 括号 、 量 词 和 预定 义 字符 类 。 

1 中 括号 

在 正则 表达 式 中 ， 中 括号 〈[]) 表示 在 一 定 的 范围 内 查找 字符 。 可 以 在 中 括号 内 设置 要 查找 的 
字符 序列 。 中 括号 在 正则 表达 式 中 起 到 至 关 重 要 的 作用 ， 在 很 多 时 候 都 要 查找 包含 某 个 范围 字符 的 
字符 串 ， 如 电话 号 码 等 。 表 8-1 列 出 了 正则 表达 式 的 中 括号 常用 的 字符 范围 。 

表 8-1 ”中 括号 常用 字符 范围 

常用 形式 功能 说 明 


[a-z] 匹配 任何 从 小 写 a~z 之 间 的 字符 [A-Za-z] | 匹配 任何 从 大 写 A 到 小 写 z 之 间 的 字符 
表 8-1 所 示 的 范围 具有 一 般 性 ， 也 可 以 使 用 [5-8] 来 匹配 从 5~8 之 间 的 数字 ， 或 者 使 用 [c-t 来 匹 
配 小 写字 母 从 c~t 之 间 的 字符 。 也 就 是 说 ， 可 以 指定 任意 的 范围 。 中 括号 的 使 用 示例 如 下 所 示 : 


<?php 

ereg('c[aeiou]t', 'I cut my hand')7 // 返 回 True 
ereg('c[aeiou]t', '‘'This crusty cat'); // 返 回 True 
ereg('c[aeiou]t', 'What cart?'); // 返 回 False 
ereg('c[aeiou]t', ‘14ct gold'); // 返 回 False 
ereg('[0-9]%', 'we are 25% complete'); // 返 回 True 
ereg('[0123456789]%'，'we are 25% complete'); // 返 回 True 
ereg('[a-z]t’', '1lith'); // 返 回 False 
ereg(' [a-z]t', 'cat'); // 返 回 True 
ereg('[a-z]t', ‘'PIT'); // 返 回 False 
ereg(' [a-zA-Z]!', '11!'); // 返 回 False 
ereg('[a-zA-Z]!', 'stop!'); // 返 回 True 
?> 


在 上 述 代码 中 , ereg0 函 数 主要 是 在 指定 的 字符 串 中 查找 相 匹 配 的 字符 或 者 字符 串 。 如 果 找 到 则 
返回 True 或 者 1， 否 则 返回 False 或 者 0。 

2. 量词 

在 中 括号 内 可 以 指定 一 定 范围 的 字符 ， 来 用 于 查找 的 对 象 。 如 果 中 括号 的 字符 和 字符 序列 有 一 
定 的 位 置 ， 那 么 在 查找 过 程 中 需要 指定 匹配 字符 串 所 在 的 位 置 ， 或 者 需要 在 字符 串 中 指定 某 一 个 字 
符 出 现 的 次 数 。 这 些 都 可 以 通过 一 个 特殊 字符 来 指定 。 这 些 特殊 字符 都 有 特定 的 含义 。 

在 正则 表达 式 开头 的 ^ 符 号 表示 它 必须 匹配 字符 串 的 开头 (或 者 更 准确 地 说 , 是 把 正则 表达 式 定 
位 在 字符 串 开 头 )。 一 个 在 正则 表达 式 末尾 出 现 的 美元 符号 (9)， 表 示 它 必须 匹配 字符 串 的 末尾 〈 也 
就 是 说 ， 把 正则 表达 式 定位 在 字符 串 的 末尾 )。 在 正则 表达 式 中 的 句点 () 匹 配 任意 单个 字符 。 如 果 想 
要 匹配 特殊 字符 中 的 某 一 个 ( 称 为 元 字符 ，metacharacter)， 需 要 使 用 反 斜 杠 对 它 进行 转 义 : 

ereg('\$5\.00', 'Your bill is $5.00 exactly'); // 返 回 True 

ereg('$5.00','Your bill is $5.00 exactly');  // 返 回 False 


正则 表达 式 默 认 区 分 大 小 写 ， 所 以 正则 表达 式 “cow” 和 字符 串 “COW” 是 不 匹配 的 。 如 果 想 


要 执行 一 个 不 区 分 大 小 写 的 POSIX 风格 的 匹配 , 可 以 使 用 eregi0 函 数 。 要 使 用 Perl 风格 的 正则 表达 
式 ， 则 可 以 使 用 preg_matchO 函 数 ， 但 是 需要 指定 一 个 标志 来 说 明 是 不 区 分 大 小 写 的 匹配 。 
常用 的 量词 含义 如 表 8-2 所 示 。 
表 8-2 常用 量词 (一 ) 


功能 说 明 功能 说 明 


p+ 匹配 任何 一 个 至 少 包 含 p 的 字符 串 匹配 任何 包含 两 个 或 三 个 p 序列 的 字符 串 
p* 匹配 任何 包含 零 个 或 多 个 p 的 字符 串 匹配 任何 至 少 包含 两 个 p 序列 的 字符 串 
匹配 任何 包含 零 个 或 一 个 p 的 字符 串 匹配 任何 以 p 结尾 的 字符 串 


匹配 任何 包含 两 个 p 序列 的 字符 串 人 

如 表 8.2 中 ， 特 殊 字 符 +、*、?、 {occurrence_range} 和 $ 标 志 都 要 跟 在 一 个 字符 序列 的 后 面 。 同 
样 还 有 一 些 特殊 字符 可 以 放 在 字符 序列 的 前 面 ， 或 者 放 在 字符 序列 的 中 间 。 这 些 特殊 字符 的 常用 形 
式 如 表 8-3 所 示 。 


表 8-3 ”常用 量词 (二 ) 


常用 形式 功能 说 明 

Pp 匹配 任何 以 p 开头 的 字符 串 

[Aa-zA-Z] 匹配 任何 不 包含 从 a~z 和 从 A~Z 的 字符 串 

pp 匹配 任何 包含 字符 p， 接 下 来 是 任何 字符 串 ， 再 接 下 来 是 p 的 字符 串 
^.{2}$ 匹配 任何 只 包含 两 个 字符 的 字符 串 

<b>(.*)</b> 匹配 任何 由 <b> 和 </b> 包 围 的 字符 串 

Pp(hp)* 匹配 任何 包含 一 个 p， 后 面 是 零 个 或 多 个 hp 的 字符 串 


当 指定 一 个 字符 类 时 ， 一 些 特殊 字符 就 失去 了 它们 的 意义 ， 同 时 其 他 字符 有 了 新 的 意义 。 特 别 
需要 注意 的 是 ， 当 ^ 字 符 不 再 是 以 前 讨论 过 的 ， 用 于 指定 以 某 个 字符 开头 ， 而 是 用 于 否定 字符 类 ( 当 
它 是 左 括号 的 第 一 个 字符 时 ) 时 ， 美 元 符号 “$” 和 句点 “.” 也 在 字符 类 中 失去 了 它们 原来 的 意义 
($” 原 指 以 某 字 符 结尾 :“.” 原 指 可 代替 任意 字符 )， 而 仅仅 是 一 个 字符 。 例 如 ，[A] 匹 配 任 一 不 是 
右 括号 的 字符 ，[$. 入 匹配 任 一 美元 符号 、 句 点 或 ^ 

不 同 的 正则 表达 式 库 为 字符 类 定义 了 不 同 的 缩写 ， 包 括 数字 、 字 母 表 的 字符 和 空白 符 。POSIX 
风格 和 Perl 风格 正则 表达 式 在 缩写 语法 上 是 有 区 别 的 。 例 如 ，POSIX 用 “[[:space:]]” 表 示 空 白 类 ， 
而 Perl 则 是 “\s”。 其 使 用 示例 如 下 所 示 : 


<?php 

ereg('^([a-z]1[0-9])', 'The quick brown fox'); // 返 回 False 
ereg('^([a-z]1[0-9])', "Jumped over'); // 返 回 True 
ereg('^([a-z]1[0-9])', "10 lazy dogs'); // 返 回 True 
ereg('catt', 'caaaaaaat'); // 返 回 True 
ereg('catt', "ct ) 7 // 返 回 False 
ereg('ca?t', "caaaaaaat') 7 // 返 回 False 
ereg('ca*t', "ct ) 7 // 返 回 False 
ereg(' [0-9]{3}j-[0-9]{3}-[0-9]{4}"， "303-555-1212") 7 // 返 回 True 
ereg( [L090t31= [0-91131= [0=9] Tt4] "64-9=555-1234")s // 返 回 False 
ereg('a (very )+big dog', 'it was a very very big dog'); // 返 回 True 
ereg('^(catldog)$', 'cat'); // 返 回 True 
ereg('^(catldog)$', 'dog'); // 返 回 True 
人 
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对 于 上 述 代码 需要 注意 的 是 字符 “|” 的 使 用 , 竖 线 () 字 符 在 正则 表达 式 中 指定 可 供 选 择 的 部 分 。 
“^catldog$” 选 自 “^cat 和 'dog$”， 意 味 着 它 匹配 以 “cat” 开 头 或 “dog” 结 尾 的 一 行 。 如 果 想 要 一 行 
只 包括 “cat” 或 “dog”， 需 要 使 用 正则 表达 式 “^(catldog)$”。 其 使 用 示例 如 下 所 示 : 


ereg('^(catldog)$', 'cat'); // 返 回 True 
ereg('^(catldog)$', 'dog'); // 返 回 True 
3. 预定 义 字 符 类 
在 PHP 的 正则 表达 式 中 ,为 了 编程 的 方便 ， 可 以 使 用 一 些 预定 义 的 字符 范围 ， 称 为 字符 类 。 字 
符 类 指定 整个 字符 范围 ， 例 如 字母 或 整数 集 。 这 些 预 定义 字符 类 如 表 8-4 所 示 。 
表 8-4 ”预定 义 字 符 类 
这 类 描述 扩展 
也 [:alnum:] 字母 和 数字 字符 [0-9a-zA-Z] 
Es [:alpha:] (letters) 字母 字符 〈 字 母 ) [a-zA-2] 
8 [:ascii:] 7 位 ASCI[ [x01-\x7F] 
a [:blank:] 水 平 空白 符 ( 空 格 、 制 表 符 ) [Wy 
区 [:cntrl:] 控制 字符 [SoLNIF] 
这 [digit:] 数字 [0-9] 
习 [:graph:] 用 墨水 打印 的 字符 〈 非 空格 、 非 控制 字符 ) [\x01-\w20] 
5 [lower] 小 写字 母 [a-z] 
册 | 可 打印 字符 《图 形 关 加 空格 和 制 表 符 ) ISONEH] 
[punct] 任意 标点 符号 ， 如 句点 (.) 和 分 号 (:) [-!"#$%&'0#+../::<=>?@MNN "OD ~] 
[:space:] 空白 (换行 、 回 车 、 制 表 符 、 空 格 、 垂 直 制 表 符 ) [mt\xoB] 
[upper] 大 写字 母 [A-Z] 
[:xdigit:] 十 六 进 制 数 字 [0-9a-fA-F] 


在 表 8-4 中 ， 每 一 个 [:something:] 类 都 可 用 于 替代 一 个 字符 类 中 的 字符 。 例 如 ， 要 查找 任 一 个 数 
字 字 符 、 大 写字 母 或 一 个 @ 符 号 ， 可 以 使 用 下 面 的 正则 表达 式 ，[@[:digit:][:upper:]]。 但 是 ， 不 能 把 
一 个 字符 类 当 作 一 个 范围 的 终点 使 用 。 


ereg('[A-[:lower:]]', 'string'); // 非 法 的 正则 表达 式 


一 些 地 区 把 某 些 字符 序列 当 作 一 个 单独 的 字符 来 考虑 一 一 它们 被 称 为 排序 序列 collating 
sequence)。 在 字符 类 中 匹配 这 些 多 字符 序列 中 的 一 个 时 ， 要 把 它 用 [ 和 .] 括 起 来 。 例 如 ， 如 果 我 们 
的 地 区 有 排序 序列 ch， 可 以 使 用 下 面 的 字符 类 来 匹配 s、t 或 ch: [st[.ch.]] 


字符 类 的 使 用 示例 如 下 所 示 : 

<?php 

ereg('^[:lower:]', 'John'); // 返 回 False 
echo ereg( '[:alnum:]', ‘jumped over'); // 返 回 True 
echo ereg( '[:digit:]'，'10 lazy dogs');  // 返 回 True 


他 之 


8.2.3 ”POSIX 正则 表达 式 函 数 


PHP 为 使 用 POSIX 风格 的 正则 表达 式 搜索 字符 串 提供 了 7 个 函数 ,分 别 为 ereg_replaceO 、eregO、 
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eregi_ replace 0 、eregiO、split0、spliti0、sql_ regcase0。 这 些 函 数 按 功 能 可 以 分 为 匹配 、 蔡 换 、 拆 分 
三 类 。 这 些 函 数 的 说 明 如 表 8-5 所 示 。 
表 8-5 ” POSIX 正则 表达 式 函 数 


名 称 语法 格式 功 能 : 
ereg0 int ereg (string $pattem. string $string [. array | 正则 表达 式 匹配 Ee 
&ySregs]) ” 
eregi0 int eregi(string $pattem. string $string [, array | 不 区 分 大 小 写 的 正则 表达 式 匹配 
人 Sregs]) 


ereg replace() | string ereg replace (string $pattem，string | 正则 表达 式 蔡 换 

S$replacement, string $string) 

eregi_replace() | string eregi replace (string $pattem，string | 不 区 分 大 小 写 的 正则 表达 式 蔡 换 
Sreplacement, string $string) 

splitO array split (string $pattern, string $string [, int 
Slimit]) 

splitiO) array spliti (string $patterm，string $string [, 
int $limit]) 

sql regcase() | string sql regcase (string $string) 


将 字符 串 分 割 到 数组 中 


正则 表达 式 不 区 分 大 小 写 将 字符 串 分 割 到 数组 中 


产生 用 于 不 区 分 大 小 的 匹配 的 正则 表达 式 


月 过 山寺 导 山 吕 小 
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1. ereg() 

该 函数 会 以 区 分 大 小 写 的 方式 在 string 中 寻找 与 给 定 的 正则 表达 式 pattem 匹配 的 子 串 。 如 果 在 
string 中 找到 pattern 模式 的 匹配 ， 则 返回 所 匹配 字符 串 的 长 度 ， 如 果 没 有 找到 匹配 或 出 错 则 返 
False。 如 果 没 有 传 入 可 选 参数 regs 或 者 所 匹配 的 字符 串 长 度 为 0， 则 返回 1。 其 使 用 示例 如 下 所 示 : 

<?php 

if(ereg('y.*e$', "'Sylvie')){ 
echo "ok"; 


表 
过 
式 


加 


} 


2 


对 于 第 三 个 可 选 参数 regs， 在 下 列 情况 使 用 ， 如 果 找到 与 pattern 中 国 括 号 内 的 子 模式 相 匹 配 的 
子 串 ， 并 且 函 数 调 用 给 出 了 第 三 个 参数 regs， 则 匹配 项 将 被 存 入 regs 数组 中 。regs[1] 包 含 第 一 个 左 
圆 括 号 开始 的 子 串 ，regs[2] 包 含 第 二 个 子 串 ， 以 此 类 推 。regs[0] 包 含 整个 匹配 的 字符 串 。 其 使 用 示 
例如 下 所 示 : 


<?php 

$date =Date ("2008-6-21"); 

if (ereg ("([0-9] {4})-([0-9] {1,2})-([0-9] {1,2})", $date, $regs)) { 
echo "$regs[3] .$regs[2] .$regs[1]"; 

} else { 
echo "Invalid date format: $date"; 

} 

$cardname = "John "Doe'"7 

$result = ereg("^[' A-Za-z]+$", $cardname, S$arr); 

echo (' 有 “，'.$result."， 个 字符 和 相 匹 配 '); 


?> 


在 上 述 代码 中 ， 主 要 判断 date、cardname 字符 串 中 是 否 匹 配 指定 的 正则 表达 式 模板 ， 并 把 匹配 
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结果 存储 到 第 三 个 参数 中 。 


2. eregi() 
该 函数 和 ereg0 函 数 完全 相同 ， 只 除了 在 匹配 字母 字符 时 忽略 大 小 写 的 区 别 。 其 使 用 示例 如 下 
所 示 : 


在 上 述 代码 中 实现 了 对 电子 邮件 的 匹配 校 验 。 
3. ereg_replace() 


该 函数 在 string 中 扫描 与 pattem 匹配 的 部 分 , 并 将 其 蔡 换 为 replacement。 返回 蔡 换 后 的 字符 串 ， 


如 果 没 有 可 供奉 换 的 匹配 项 则 返回 原 字符 串 。 其 使 用 示例 如 下 所 示 : 


在 该 函数 的 使 用 中 ， 如 果 pattern 字符 串 中 包含 了 一 个 括号 ， 则 replacement 可 以 包含 形 如 Ndigit 
的 子 串 ， 这 些 子 串 将 被 蔡 换 为 数字 表示 的 第 几 个 括号 内 的 子 串 ; 0 则 包含 了 字符 串 的 整个 内 容 ，\1 
是 第 一 个 成 功 匹 配 的 子 串 ， 最 多 可 以 用 9 个 子 串 。 括 号 可 以 嵌 套 ， 此 情形 下 以 左 圆 括号 来 计算 顺序 。 


如 果 未 在 string 中 找到 匹配 项 ， 则 string 将 原样 返回 。 其 使 用 示例 如 下 所 示 : 


4. eregi_replace() 
该 函数 和 ereg_replace0 函 数 完 全 相同 ， 只 除了 在 匹配 字母 字符 时 忽略 大 小 写 的 
例如 下 所 示 : 


区 别 。 其 使 用 示 


echo Svar7 

在 上 述 代码 中 ， 会 在 string 中 搜索 空格 ， 并 把 多 个 空格 蔡 换 为 一 个 空格 。 

5. split() 

该 函数 返回 一 个 字符 串 数组 ， 每 个 单元 为 string 经 区 分 大 小 写 的 正则 表达 式 pattern 作为 边界 分 
割 出 的 子 串 。 如 果 设 定 了 limit， 则 返回 的 数组 最 多 包含 limit 个 单元 ， 而 其 中 最 后 一 个 单元 包含 了 
string 中 剩余 的 所 有 部 分 。 如 果 出 错 ， 则 splitO 函 数 返回 False。 其 使 用 示例 如 下 所 示 : 


<?php 

// 分 隔 符 可 以 是 斜 线 、 点 ， 或 横 线 

$date = "04/30/2008"™; 

list($month, $day, $year) = split ('[/.-]', $date); 

echo "Month: $month; Day: $day; Year: $year<br />\n"; 

> 

在 上 述 代码 中 ， 会 把 日 期 字符 串 date 以 字符 “/” 或 “-” 分 割 存 储 到 数组 中 。 

6. spliti() 

该 函数 和 split0 函 数 相同 ， 只 除了 在 匹配 字母 字符 时 忽略 大 小 写 的 区 别 。 其 使 用 示例 如 下 所 示 : 

<?php 

$string = "aBBBaCCCADDDaEEEaGGGA"; 

$chunks = spliti ("a", $string, 5); 

print r($chunks); 

2 

在 上 述 代 码 中 ， 以 字符 “a” 作 为 分 隔 符 ， 把 分 割 出 来 的 字符 存储 到 数组 chunks 中 ， 然 后 输出 
所 有 的 元 素 ， 其 中 5 表示 数组 的 大 小 。 

7. Sql_regcase() 

该 函数 返回 与 string 相 匹配 的 正则 表达 式 ， 不 论 大 小 写字 母 。 返 回 的 表达 式 是 将 string 中 的 每 
个 字母 字符 转换 为 方 括号 表达 式 , 该 方 括号 表达 式 包含 了 该 字母 的 大 小 写 形式 。 其 他 字符 保留 不 变 。 
其 使 用 示例 如 下 所 示 : 

<?php 

echo sql regcase ("Foo - bar.") 7 

Ea 


上 述 代 码 执行 完毕 后 ,会 输出 与 该 字符 串 匹配 的 正则 表达 式 , 如 “[Ff[Oo][O0] - [BbJ[Aa][Rr]”。 
8.2.4 ”Perl 正则 表达 式 语法 


一 直 以 来 ，Perl 被 视 为 强大 的 正则 表达 式 的 标准 ， 它 提供 了 一 种 全 面 的 正则 表达 式 语 言 ， 即 使 
是 最 复杂 的 字符 串 模式 ， 也 可 以 被 这 种 正则 表达 式 语 言 搜索 和 蔡 换 。PHP 使 用 一 个 被 称 为 pcre 的 C 
库 ， 几 乎 完全 支持 Perl 正则 表达 式 的 特性 。Perl 风格 的 正则 表达 式 和 POSIX 类 似 。 实 际 上 ，Perl 的 
正则 表达 式 语 法 就 是 由 POSIX 实现 派生 的 ， 二 者 有 很 多 相似 的 地 方 。 

Perl 风格 的 正则 表达 式 模仿 Perl 模式 的 语法 ， 即 每 个 模式 都 必须 用 一 对 分 隔 符 括 起 来 。 习 惯 上 
使 用 左 斜 杠 /)， 例 如 ，/patterm/。 但 是 ， 任 意 非 数 字 字 母 的 字符 (除了 反 和 斜 杠 (\))〉 都 可 用 于 分 隅 一 
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个 Perl 风格 的 模式 ， 这 在 匹配 包含 斜 杠 的 字符 串 时 很 有 用 ， 如 文件 名 。 例如 ， 下 面 的 语句 是 等 效 的 : 


preg match('/\/usr\/local\//', "/usr/local/bin/per1')7 // 返 回 True 
preg match('#/usr/local/#', '/usr/local/bin/perl'); // 返 回 True 
小 括号 (0)、 大 括号 ({)、 中 括号 () 和 尖 括 号 (<>) 可 被 作为 模式 分 隔 符 使 用 : 
Preg_match('{/usr/local/}", /usr/local/bin/perl'): /返回 True 


虽然 Perl 正则 表达 式 语 法 包含 POSIX 结构 , 但 在 Perl 中 一 些 模式 组 件 有 不 同 的 意义 。 Perl 正则 
表达 式 特 别 为 单行 文字 匹配 进行 了 优化 (虽然 有 一 些 选项 可 改变 这 个 行为 )。 句 点 () 匹 配 除 换行 符 Cm) 
之 外 的 任意 字符 。 美 元 符号 ($) 匹 配 字 符 串 的 末尾 或 在 换行 符 之 前 以 换行 符 结尾 的 字符 串 。 

preg match('/is (.*)$/', "the key is in my pants", $captured);// $captured[1] 是 'in 

my pants’' 


Perl 风格 的 正则 表达 式 同 POSIX 正则 表达 式 一 样 ， 都 可 以 通过 量词 构建 复杂 的 字符 串 ， 如 下 
所 示 : 

/fo+/ 

/f0{2,4}/ 


第 一 段 代码 表示 匹配 fo 后 面 跟 一 个 或 多 个 字符 的 字符 串 ， 如 food、fool 和 fo4。 第 二 行 代码 表 
示 匹 配 革 后 面 跟 有 2 一 4 个 o， 如 fool、fooool 和 foosball 等 。 

1， 修 饰 符 

在 进行 PHP 正则 表达 式 匹配 的 过 程 中 ， 可 能 会 有 不 同 的 要 求 ， 如 不 区 分 大 小 写 的 搜索 ， 或 忽略 
语法 中 的 注释 。 这 些 调整 称 为 修饰 符 ， 修 饰 符 对 编写 简洁 短小 的 表达 式 有 很 大 的 帮助 。 表 8-6 列 出 
了 常用 的 修饰 符 。 


表 8-6 ”常用 修饰 符 
修饰 符 描述 
i 不 区 分 大 小 写 进 行 匹配 
m 将 一 个 字符 串 视 为 多 行 。 默认 情况 下 , ^ 和 $ 字 符 匹 配 字符 串 中 的 最 开始 和 最 末尾 。 使 用 m 修饰 符 将 使 ^ 
和 $ 匹 配 字符 串 中 每 行 的 开始 
Ss 将 一 个 字符 串 视 为 一 行 ， 忽 略 其 中 的 换行 符 ， 它 与 m 修饰 符 正好 相反 
4 忽略 正则 表达 式 中 的 空白 和 注释 
U 第 一 次 匹配 后 停止 ， 即 查找 到 第 一 个 字符 串 后 ， 停 止 后 面 的 搜索 
表 中 的 修饰 符 可 以 直接 放 在 正则 表达 式 的 后 面 ， 其 示例 如 下 所 示 : 
/wmd/i 
/taxation/si 


在 上 述 代码 中 ， 第 一 行 代码 表示 可 以 匹配 WMD、wMd、WMd、wmd 或 者 字符 串 wmd 的 任何 
其 他 大 小 写 的 形式 。 第 二 行 代码 表示 不 区 分 大 小 写 ， 并 忽略 字符 串 中 换行 符 。 

2. 元 字符 
利用 Perl 正则 表达 式 还 可 以 使 用 各 种 元 字符 来 搜索 匹配 。 元 字符 就 是 一 个 前 面 有 反 斜 线 的 字母 
字符 ， 表 示 某 种 特殊 含义 。Perl 常用 的 元 字符 如 表 8-7 所 示 。 


i 


表 8-7 ”常用 元 字符 


元 字 符 描 述 

\A 只 匹配 字符 串 开 头 

wb 匹配 单词 边界 2 

B 匹配 除 单词 边界 外 的 任意 字符 ee 

\d 匹配 数字 字符 ， 它 与 [0-9] 相 同 9 

D 匹配 非 数 字 字 符 

国 匹配 空白 字符 

\S 匹配 非 空 白字 符 

\w 匹配 任何 只 包含 下 划 线 和 字母 数字 字符 的 字符 串 。 它 与 [a-zA-Z0-9] 相 同 

\W 匹配 没有 下 划 线 和 字母 数字 字符 的 字符 串 第 
元 字符 的 使 用 示例 如 下 所 示 : 
/sa\b/ hs - 
/\blinux\b/i 2 符 
/sa\B/ 3 串 
/\$dt/g 可 和 

正 


=] 


第 1 行 代码 表示 单词 边界 定义 在 字符 串 的 右 侧 ， 所 以 这 将 匹配 pisa 和 lisa 等 ， 但 不 能 匹配 sand。 
第 2 行 代码 会 返回 linux 的 第 一 次 出 现 ， 不 区 分 大 小 写 。 在 第 3 行 代码 中 ，\B 与 单词 边界 元 字符 相反 ， 
匹配 除 单词 边界 之 外 的 任意 字符 ， 这 将 匹配 sand 和 Sally 等 字符 串 ， 而 不 能 匹配 Melissa。 第 4 行 代码 
表示 返回 满足 以 下 条 件 的 字符 串 的 所 有 实例 :字符 串 包含 一 个 美元 符号 ， 后 面 跟 一 个 或 多 个 数字 。 


8.2.5 ”Perl 正则 表达 式 函 数 


表 
送 
式 


有 5 类 函数 可 用 于 Perl 兼容 正则 表达 式 : 匹配 、 蔡 换 、 拆 分 、 过 滤 和 引用 文本 的 通用 函数 。 它 
们 分 别 为 : preg_grepO、Ppreg_matchO、preg_match_all0、Ppreg_quoteO0、Ppreg_IeplaceO0、Ppreg_replace 
callback()、preg_split0。 其 详细 信息 如 表 8-8 所 示 。 
表 8-8 ”Perl 正 则 表达 式 函 数 


名 称 功 能 
preg_grep() array preg_grep (string $pattern, array $input [. int | 返回 与 模式 匹配 的 数组 单元 
Sflags]) 
preg_matchO int preg_match (string $pattermm. string $subject [. | 进行 正则 表达 式 匹配 
array $matches [. int $flags]]) 
preg_match all0 | int preg_match all (string $pattern, string $subject | 进行 全 局 正则 表达 式 匹 配 
array $matches [. int $flags]) 
preg_quote() string preg_quote (string $str [. string $delimiter]) 用 于 转 义 正则 表达 式 字符 , 即 函数 创建 一 个 
只 匹配 给 定 字符 串 的 正则 表达 式 
preg_replace() mixed preg replace (mixed S$pattern.、 mixed | 执行 正则 表达 式 的 搜索 和 蔡 换 
Sreplacement. mixed $subject [. int $limit]) 
preg_replace mixed preg_replace_callback (mixed S$patterm. | 用 回调 函数 执行 正则 表达 式 的 搜索 和 蔡 换 
_callback0) callback $callback. mixed $subject [. int $limit]) 
preg_splitO array preg_split (string $pattern. string $subject [. | 用 正则 表达 式 分 割 字符 串 


int $limit [, int $flags]]) 
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1. preg_grep() 

函数 返回 一 个 数组 ， 其 中 包括 了 input 数组 中 与 给 定 的 pattem 模式 匹配 的 单元 。flags 可 以 是 标 
记 PREG GREP INVERT， 如 果 传 入 此 标记 ， 该 函数 会 返回 输入 数组 中 不 匹配 给 定 pattern 的 单元 。 
该 函数 的 使 用 示例 如 下 所 示 : 

<?php 

$foods=array ("apple", "orange", "pip", "banana"); 

$fo0d=preg grep("/^p/",$fo0ds); 


print r($food); 
Rs 


在 上 述 代 码 中 创建 了 一 个 数组 ， 里 面包 含 了 4 个 相应 的 数组 元 素 。/^p/ 模 式 表示 匹配 以 字符 p 
开始 的 字符 串 ，foods 数组 表示 要 搜索 的 字符 串 对 象 ， 这 里 要 对 每 个 数组 元 素 进行 匹配 。 该 示例 的 输 
出 结果 为 “Array([2]=>pip)”。 

2. preg_match() 

在 该 函数 中 ， 从 subject 字符 串 中 搜索 与 pattem 给 出 的 正则 表达 式 匹配 的 内 容 。preg_matchO 函 
数 返 回 pattem 匹配 的 次 数 ，0 次 (没有 匹配 ) 或 1 次 ， 因 为 preg_matchO 函 数 在 第 一 次 匹配 之 后 将 
停止 搜索 。preg_match_all0 函 数 则 相反 ， 会 一 直 搜索 到 subject 的 结尾 处 。 如 果 出 错 则 返回 False。 
其 使 用 示例 如 下 所 示 : 


<?php 
preg match('/y.*e$/', 'Sylvie'); // 返 回 True 
preg match('/y.*e$/i','SyLvIe'); // 返 回 True 


二 


在 上 述 代 码 中 ， 是 以 区 分 大 小 写 和 不 区 分 大 小 写 两 种 形式 进行 匹配 的 。 如 果 提 供 了 matches， 
则 其 会 被 搜索 的 结果 所 填充 。matches[0] 将 包含 与 整个 模式 匹配 的 文本 ，matches[1] 将 包含 与 第 一 个 
捕获 的 括号 中 的 子 模式 所 匹配 的 文本 ， 以 此 类 推 。flags 可 以 标记 PREG_ OFFSET CAPTURE。 如 果 
设 定 该 标记 ， 对 每 个 出 现 的 匹配 结果 也 同时 返回 其 附属 的 字符 串 偏 移 量 。 注 意 这 改变 了 返回 的 数组 
的 值 ， 使 其 中 的 每 个 单元 也 是 一 个 数组 ， 其 中 第 一 项 为 匹配 字符 串 ， 第 二 项 为 其 偏 移 量 。 该 标记 自 
PHP 4.3.0 起 可 用 。 其 使 用 示例 如 下 所 示 : 


<?php 
preg match('/y(.*)e$/', 'Sylvie', $m); 
echo $m[0]; 
echo "<br>"; 
echo $m[1]; 
?> 


在 上 述 代码 中 ,会 把 搜索 到 的 字符 串 存储 在 数组 m 中 。 其 执行 结果 为 ,数组 的 第 一 个 元 素 存储 
的 是 整个 “Sylvie”， 数 组 的 第 二 个 元 素 是 “lvi”。 


如 果 只 起 查看 一 个 字符 囊 是 否 包 合 在 另 一 个 字符 囊 中 ， 不 要 用 preg_matchO， 函 数 可 以 用 stposO 函 数 
提示 [ 或 Ustr0 函 数 替 代 ， 这 会 快 得 多 。 


3. preg_match_all() 
该 函数 表示 在 subject 中 搜索 所 有 与 pattem 给 出 的 正则 表达 式 匹 配 的 内 容 ， 并 将 结果 以 flags 指 
定 的 顺序 放 到 matches 中 。 搜 索 到 第 一 个 匹配 项 之 后 ， 接 下 来 的 搜索 从 上 一 个 匹配 项 末尾 开始 。 
flags 可 以 是 下 列 标记 的 组 合 , 注意 把 PREG PATTERN ORDER 和 PREG SET ORDER 合 起 来 
使 用 没有 意义 。 其 详细 信息 如 表 8-9 所 示 。 
表 8-9 flags 标 记 组 合 
名 称 含义 


PREG PATTERN ORDER | 表示 对 结果 排序 使 matches[0] 为 全 部 模式 匹配 的 数组 ，matches[1] 为 第 一 个 括号 中 的 
子 模式 匹配 的 字符 串 组 成 的 数组 ， 以 此 类 推 


PREG SET_ ORDER 表示 对 结果 排序 使 matches[0] 为 第 一 组 匹配 项 的 数组 , matches[1] 为 第 二 组 匹配 项 的 
数组 ， 以 此 类 推 
该 函数 的 使 用 示例 如 下 所 示 : 
<?php 


// \\2 是 一 个 逆向 引用 的 例子 ， 其 在 PCRE 中 的 含义 是 
// 必须 匹配 正则 表达 式 本 身 中 第 二 组 括号 内 的 内 容 ， 本 例 中 
// 就 是 ([\w]+) 。 因 为 字符 串 在 双 引 号 中 ， 所 以 需要 
// 多 加 一 个 反 斜 线 
$html = "<b>bold text</b><a href=howdy.html>click me</a>"7 
preg match all ("/(<([\w]+) [^>]*>) (.*) (<\/\\2>)/", $html, $matches); 
for ($i=0; $i< count ($matches[0]); $i++) { 
echo "matched: ".Smatches[0] [$i]."<br>"; 
echo "part 1: ".$matches[1] [$i]."<br>"; 
echo "part 2: ".$matches[3] [$i]."<br>"; 
echo "part 3: " .Smatches [4] [$i]."<br><br>"; 


?> 

在 上 述 代码 中 创建 了 一 个 正则 表达 式 , 用 来 匹配 一 个 含有 HTML 标记 的 字符 串 ， 如 果 匹 配 成 功 
则 把 匹配 的 字段 存储 到 数组 matches 中 。 

4. preg_quote() 

该 函数 以 str 为 参数 并 给 其 中 每 个 属于 正则 表达 式 语 法 的 特殊 字符 前 面 加 上 一 个 反 斜 线 。 如 果 
需要 以 动态 生成 的 字符 串 作 为 模式 去 匹配 ， 则 可 以 用 此 函数 转 义 其 中 可 能 包含 的 特殊 字符 。 如 果 提 
供 了 可 选 参数 delimiter (用 于 设 定 正则 表达 式 的 定 界 符 )， 该 字符 也 将 被 转 义 。 可 以 用 来 转 义 PCRE 
函数 所 需要 的 定 界 符 ， 最 常用 的 定 界 符 是 斜 线 /。 其 使 用 示例 如 下 所 示 : 

<?php 

$keywords = '$40 for a g3/400°'; 

$keywords = preg quote ($keywords, '/'); 


echo $keywords; // returns \$40 for a g3\/400 
> 


在 上 述 代码 的 字符 串 keywords 中 ， 字 符 “$” 和 字符 “/” 是 特殊 字符 ， 在 使 用 该 函数 时 ， 会 在 
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这 些 特殊 字符 前 面 加 上 反 斜 线 。 正 则 表达 式 的 特殊 字符 有 : 3$、^ 人 、*、0、=、+、 修 、 [中 |、 、 一。 
5. preg_replace() 
该 函数 的 执行 流程 是 在 subject 对 象 中 搜索 符合 pattem 模式 的 字符 串 匹 配 项 ， 并 将 搜索 到 的 字 
符 串 蔡 换 为 replacement。 如 果 指 定 了 limit， 则 仅 蔡 换 limit 个 匹配 ， 如 果 省 略 limit 或 者 其 值 为 -1， 
则 所 有 的 匹配 项 都 会 被 蔡 换 。 其 使 用 示例 如 下 所 示 : 


<?php 
$better = preg replace('/<.*?>/', '!', "do <b>not</b> press the button'); 人 


scontractions = array("/dont/i" “/won't/i"e ™“/can t/i")s 
S$expansions = array('do not', "will not', "can not'); 

$string = "Please don't yellIl can't jump while you won't speak"; 
$longer = preg_ replace($contractions, $expansions, $string); 2 
echo $better; 

echo "<br>"; 

echo $longer; 

2 


在 上 述 代码 中 ,第 1 行 代码 表示 把 最 后 一 个 字符 串 参数 中 <b> 或 <b> 字 符 ， 蔡 换 为 字符 “!”。 在 
第 2 行 代码 中 ， 发 现 pattern 和 replacement 还 可 以 是 数组 ， 那 么 这 时 函数 将 循环 处 理 每 个 数组 的 每 
个 元 素 , 一 旦 找到 就 会 蔡 换 。 上 述 代 码 的 执行 结果 为 : “do !not! press the button” 和 “Please do not yellI 
can not jump while you will not speak ”。 

6. preg_replace_callback() 

该 函数 的 行为 几乎 和 preg_replace0 函 数 一 样 ,都 是 进行 搜索 和 蔡 换 ,但 不 是 提供 一 个 replacement 
参数 ， 而 是 指定 一 个 callback 函数 。callback 函数 将 以 目标 字符 串 中 的 匹配 数组 作为 输入 参数 ， 并 返 
回 用 于 蔡 换 的 字符 串 。 其 使 用 示例 如 下 所 示 : 

<?php 

function titlecase ($s) { 
return ucfirst (strtolower ($s[0])); 

l $string = "goodbye cruel world'; 

Snew = preg replace callback('/\w+/', 'titlecase', $string); YL 


echo $new; 
> 


在 上 述 代 码 中 创建 了 一 个 函数 titlecase0， 其 功能 是 把 参数 s 的 第 一 个 字母 转换 成 大 写字 母 。 在 
第 1 行 中 ， 在 字符 串 string 中 搜索 与 字符 串 “Aw+/” 匹 配 的 字符 串 ， 如 果 找 到 ， 就 会 调用 上 面 创建 
的 函数 titlecase0， 并 把 找到 的 匹配 字符 串 传 给 该 函数 。 

7. preg_split() 

该 函数 返回 一 个 数组 , 包含 的 是 在 字符 串 subject 中 以 pattern 字符 串 匹配 的 边界 所 分 割 的 子 串 。 
如 果 指 定 了 limit， 则 最 多 返回 limit 个 子 串 ， 如 果 limit 是 -1， 则 意味 着 没有 限制 ， 可 以 用 来 继续 指 
定 可 选 参数 flags。 

flags 可 以 是 标记 的 任意 组 合 〈 用 按 位 或 运算 符 组 合 )， 其 标记 信息 如 表 8-10 所 示 。 


表 8-10 ”flags 标 记 组 合 
名 称 x 
PREG SPLIT NO EMPTY 如 果 设 定 了 该 标记 ， 则 preg_split0 函 数 只 返回 非 空 的 成 分 
PREG SPLIT DELIM CAPTURE | 如 果 设 定 了 该 标记 ， 定 界 符 模式 中 的 括号 表达 式 也 会 被 捕获 并 返回 
PREG SPLIT_ OFFSET CAPTURE | 如 果 设 定 了 该 标记 , 对 每 个 出 现 的 匹配 结果 也 同时 返回 其 附属 的 字符 串 偏 移 
量 。 注 意 这 改变 了 返回 的 数组 的 值 ， 使 其 中 的 每 个 单元 也 是 一 个 数组 ， 其 中 
第 一 项 为 匹配 字符 串 ， 第 二 项 为 其 在 subject 中 的 偏 移 量 


该 函数 的 使 用 示例 如 下 所 示 : 
<?php 
$ops = preg_ split('{[+*/-]}', '3+5*9/2°'); 
$opsl = preg split('{([+*/-])}', '3+5*9/2', -1, PREG SPLIT DELIM CAPTURE); 
print r($ops); 
echo "<br>"; 
print r($ops1); 
> 


在 上 述 代 码 中 ， 以 指定 的 分 隔 符 “+”“*”“/” 和 “-” 拆 分 字符 串 “3+5*9/2”， 并 把 拆 分 后 
的 字符 以 数组 的 形式 存储 到 ops 和 opsl 中 。 这 里 需要 注意 的 是 , 对 opsl 的 拆 分 使 用 到 了 flags 参数 。 


8.3 ”普通 字符 串 函数 


除了 前 面 介绍 的 基于 正则 表达 式 的 函数 外 , 还 有 一 类 普通 的 字符 串 函 数 , 如 获取 字符 串 的 长 度 、 
字符 串 的 比较 等 。 这 类 函数 有 100 多 个 ， 用 来 处 理 字符 串 的 各 个 方面 。 本 节 将 对 常用 的 字符 串 函 数 
进行 介绍 。 


8.3.1 获取 字符 串 长 度 


获取 一 个 字符 串 的 长 度 , 在 PHP 程序 中 应 用 得 非常 普遍 ， 如 判断 客户 端 输 入 密码 的 长 度 ， 获 取 
客户 端 提交 的 留言 信息 的 长 度 等 。 

获取 字符 串 长 度 的 函数 是 sttlen0， 其 语法 格式 如 下 所 示 : 

int strlen ( string $string ) 

该 函数 的 返回 值 为 整 型 ， 表 示 字 符 串 string 的 长 度 ， 如 果 返 回 值 为 0， 则 表示 该 字符 串 为 空 。 
其 使 用 示例 如 下 所 示 : 

<?php 

function len($str){ 
if(strlen($str)>6 and strlen(Sstr)<12) 
echo "密码 长 度 符合 "; 


else 


echo "密码 过 短 或 者 过 长 "; 


ES 


汝 


表 
藉 
式 


len("java603.”) 7 
ee 


在 上 述 代码 中 使 用 stlen0 函 数 获取 字符 串 的 长 度 。 如 果 一 个 字符 串 中 包含 一 个 空格 ， 空 格 同样 
也 被 计算 在 内 。 
8.3.2 ”字符 串 比 较 


比较 两 个 字符 串 的 大 小 , 在 任何 语言 中 都 非常 常见 。 在 PHP 中 ,提供 了 4 个 函数 进行 字符 串 的 
比较 ， 分 别 为 strecmpO、strcasecmpO、strspnO、strcspnO。 其 详细 信息 如 表 8-11 所 示 。 
表 8-11 ”字符 串 比较 函数 
功 能 


名 称 语法 格式 
stremp() int stremp (string $strl, string $str2) 执行 字符 串 的 大 小 比较 ， 并 区 分 大 小 写 


stcasecmp0 进行 字符 中 比较 ， 不 区 分 大 小 写 


strspnO int strspn (string $strl. string $str2 [, int $start | 返回 一 个 字符 串 包含 另外 一 个 字符 串 的 第 一 部 分 
[intSlengtb]) 长 度 


strcspnO int strcspn (string $strl. string $str2 [, int $start | strcspn 函数 返回 strl 中 包含 str2 中 所 没有 字符 的 第 
[int $length]]) 一 部 分 的 长 度 
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1. strcmp() 
在 该 函数 中 ， 如 果 参 数 strl 小 于 参数 srt2， 则 返回 小 于 0 的 值 ， 如 果 参 数 strl 大 于 参数 str2， 
则 返回 大 于 0 的 值 ， 如 果 两 个 参数 相同 ， 则 返回 值 为 0。 其 使 用 示例 如 下 所 示 ; 


<?php 
$psdl="java603"; 
$psd2="java6 "7 
if(strcmp ($psdl, $psd2)==0) { 
echo "密码 和 确认 密码 相同 "; 
1 
elsef{ 
echo "密码 和 确认 密码 不 同 ， 请 重新 输入 "; 
} 


?> 


2. strcasecmp() 
该 函数 的 功能 和 stremp0 函 数 基本 相同 。 除 了 比较 时 区 分 大 小 写 。 其 使 用 示例 如 下 所 示 : 


<?php 
$varl = "Hello"; 
$var2 = "hello"; 


if (strcasecmp ($varl, $var2) == 0) { 
echo ' 这 两 个 字符 串 在 不 区 分 大 小 写 的 情况 下 相同 '; 


2 


3. strspn() 
该 函数 的 作用 是 匹配 并 返回 第 二 个 参数 中 的 单字 符 连续 出 现 长 度 的 值 ， 它 的 第 三 个 参数 和 第 四 
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个 参数 可 选 ， 分 别 表 示 开 始 匹配 的 位 置 和 返回 的 最 大 值 。 其 使 用 示例 如 下 所 示 : 


<?php 

S$var = strspn("42123 is the answer", "1234567890"); 
echo $var; // 显 示 5 

echo strspn("faoaaol"， "oa"，1，3); // 显示 3 

echio Stropa("too", "gm. 1 2)7 /7 旺 示 2 

echo strspn("l2abcd","123456"); 

2 


4. strcspn() 
strcspnO 函 数 的 使 用 示例 如 下 所 示 : 


<?php 
$password="al2345"; 
echo strcspn($password,"1234567890"); 
if(strspn($password,"1234567890")==0){ 
echo "该 密码 中 包含 了 非 数 字 字 符 "; 
} 


2 


8.3.3 ”字符 串 大 小 写 转换 


在 PHP 中 , 有 4 个 函数 处 理 字符 串 中 大 小 写 转换 的 问题 。 它们 分 别 为 : strtolowerO 、strtoupperO、 


ucfirst0 和 ucwords()。 其 详细 信息 如 表 8-12 所 示 。 
表 8-12 大 小 写 转换 函数 


名 称 功 能 
strtolowerO 把 字符 串 的 字母 全 部 转换 为 小 写字 母 
strtoupperO 把 字符 囊 的 字母 全 部 转换 为 大 写字 母 
ucfirstO) 把 字符 串 的 第 一 个 字母 转换 为 大 写 ， 非 字母 字符 不 受 影响 


ucwords() 把 字符 串 的 每 个 单词 的 首 字母 转换 为 大 写字 母 ， 非 字母 字符 不 受 
影响 


1. strtolower() 
该 函数 的 返回 值 是 转换 后 的 字符 串 ， 非 字母 字符 不 受 影响 。 其 使 用 示例 如 下 所 示 : 


<?php 
Sname="This Is a dog"; 
echo strtolower ($name); 
区 


2. strtoupper() 
该 函数 的 返回 值 是 转换 后 的 字符 串 。 其 使 用 示例 如 下 所 示 : 
<?php 


Sname="china is very big"; 
echo strtoupper ($name); 
?2> 


ES 


注 


表 
达 
式 
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3. ucfirst() 

该 函数 只 处 理 字符 串 的 第 一 个 字母 ， 其 返回 值 是 转换 后 的 字符 串 。 其 使 用 示例 如 下 所 示 : 

<?php 

echo ucfirst ("what a beautiful day today!");// 显 示 What a beautiful day today! 

> 

4. ucwords() 

该 函数 在 转换 过 程 中 , 以 空格 来 界定 是 否 是 单词 ,“today!Hi”“today.Hi” 会 被 认为 是 一 个 单词 ， 
每 个 单词 的 首 字母 外 的 其 余 字 母 的 大 小 写 状态 并 不 改变 。 其 使 用 示例 如 下 所 示 : 

<?php 


echo ucwords ("what a beautiful day today!");// 显 示 What A Beautiful Day Today! 
Re 


8.3.4 字符 串 与 HTML 相互 转换 


把 字符 串 或 整个 文件 转换 为 适合 于 Web 形式 的 文本 信息 ， 在 PHP 中 可 以 借助 函数 来 完成 这 种 
功能 。 这 类 函数 按 功 能 可 以 分 为 两 类 ， 一 类 是 将 纯 文本 转换 为 HTML， 一 类 是 将 HTML 转换 为 纯 
文本 。 

1. 将 纯 文本 转换 为 HTML 

把 纯 文本 转换 为 HTML 的 信息 ， 需 要 使 用 下 面 这 些 函 数 。 其 详细 信息 如 表 8-13 所 示 。 

表 8-13 ” 纯 文本 转换 HTML 函 数 


名 称 语法 格式 功 能 
nl2brO 在 字符 串 中 插入 一 个 HTML 标记 


htmlentities() string htmlentities (string $string [, int $quote_style [, | 转换 字符 为 HTML 字符 编码 
string $charset]]) 
htmlspecialchars() string htmlspecialchars (string $string [, int $quote_style | 将 特殊 字符 转 成 HTML 格式 
[. string $charset]]) 
array get_html translation table ([int S$table [. int | 返回 可 以 转换 的 HTML 实体 
S$quote_style]]) 
string strtr (string $str. string $from. string $to) 或 字符 串 替 换 
String strtr (string $str, array $replace_pairs) 
该 函数 会 把 字符 串 中 的 换行 符 “\n” 蔡 换 成 “<br />”， 并 返回 修改 后 的 字符 串 。 该 函数 的 使 用 
示例 如 下 所 示 : 
<?php 


echo nl2br ("foo isn't\n bar"); // 显示 ( 源 代码 中 ) foo isn't<br /> bar 
> 


get_html translation 
_table0 


StrtrO 
(1) nl2brO 


(2) htmlentitiesO 
该 函数 可 以 将 字符 串 中 一 些 字符 转换 为 HTML 实体 ， 默 认 情况 下 主要 包括 这 4 个 字符 :“<”、 


“>””“&@” 和 “”， 分 别 转 换 为 HTML 实体 为 : “<” “>”“&” 和 “"”。 


htmlentitiesO 函数 的 第 二 个 可 选 参数 可 以 选择 引号 的 转换 模式 ， 可 以 选择 三 个 常量 : 
ENT_COMPAT 表示 对 双 引 号 进行 编码 ,不 对 单 引号 进行 编码 , ENT_QUOTES 表示 同时 对 单 引 号 和 
双 引 号 进行 编码 ，ENT_NOQUOTES 表示 两 个 都 不 转换 ， 默 认 值 为 ENT_COMPAT。 

第 三 个 可 选 参数 表示 所 转换 字符 的 编码 集 。 其 常用 编码 如 表 8-14 所 示 。 


表 8-14 “字符 编码 集 oo 
字符 集 描述 
ISO-8859-1 ”| ISO8859-1 西欧 ，Latin-1 
ISO-8859-15 | ISO8859-15 西欧 ，Latin-9。 增 加 了 Latin-1 (ISO-8859-1) 中 缺少 的 欧元 
符号 、 法 国 及 芬兰 字母 
UTF-8 ASCII 兼容 多 字 节 8-bit Unicode 
cp866 ibm866. 866 DOS- 特 有 的 Cyrillic 字母 字符 集 。PHP 4.3.2 开始 支持 该 字 第 
符 集 
cp1251 Windows-1251,win-1251, 1251 Windows- 特 有 的 Cyrillic 字母 字符 集 。PHP 4.3.2 开始 支持 这 
该 字符 集 字 
cp1252 Windows-1252, 1252 Windows 对 于 西欧 特有 的 字符 集 符 
KOI8-R koi8-m, koi8r 俄 文 。 PHP 4.3.2 开始 支持 该 字符 集 加 
BIG5 950 繁体 中 文 ， 主 要 用 于 中 国 台湾 正 
GB2312 936 简体 中 文 ， 国 际 标准 字符 集 则 
BIGS-HKSCS 繁体 中 文 ，Bigs 的 延伸 ， 主 要 用 于 中 国 香港 法 
Shift_JIS SJIS., 932 日 文 式 
EUC-JP EUCJP 日 文 
该 函数 的 使 用 示例 如 下 所 示 : 
<?php 


$str = "A 'quote' is <b>bold</b>"; 

echo htmlentities ($str); // 转 换 结果 为 : A 'quote' is &lt;bggt;boldglt;/bggt; 
echo htmlentities ($str，ENT_QUOTES); // 转换 结果 为 : A &#039;quote&#039; is &lt; 
b&gt;bold&glt;/bg&gt7 

2 


将 上 述 代 码 执 行 后 ， 可 能 看 不 到 上 面 的 转换 结果 ， 这 时 在 显示 该 结果 的 网 页 中 ， 选 择 【 查 
看 】|【 源 文件 】 命 令 ， 就 会 看 到 注释 中 显示 的 结果 。 

(3) htmlspecialchars() 

该 函数 将 特殊 字符 转 成 HTML 的 字符 串 格式 (&.…:)。 最 常用 到 的 场合 可 能 是 处 理 客户 留言 的 留 
言 板 。 此 函数 只 转换 下 面 的 特殊 字符 , 分 别 是 :“&”( 和 ) 转 成 “&amp”“"”( 双 引号 ) 转 成 “&quot;”、 
“<”( 小 于 ) 转 成 “&lt;”“>”( 大 于 ) 转 成 “&gt”。 对 于 该 函数 的 第 二 个 参数 和 第 三 个 参数 的 含义 ， 
同 htmlentities0 函 数 中 的 参数 含义 相同 。 该 函数 的 使 用 示例 如 下 所 示 : 

<?php 

Snew = htmlspecialchars ("<a href='test'>Test</a>", ENT QUOTES); 


echo $new; // &lt;a href=&g#039;testg#039;&gt;Testglt;/aggt; 
Sy 


查看 该 函数 的 输出 结果 和 上 面 函 数 的 查看 方式 相同 。 
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如 果 gethtmlspecialchars0 要 与 nl2br0 等 函数 结合 使 用 ， 应 当 在 gethtmlspecialcharsO) 函 数 之 后 执行 
提示 nl2br0; 否则 ，nl2brO 函 数 生 成 的 <br/> 函 数 标记 将 被 转换 为 可 见 字符 。 


(4) get html translation tableO 

该 函数 有 两 个 常量 参数 ， 第 一 个 参数 表示 选择 显示 哪 种 转换 模式 下 的 内 容 ，HTML ENTITIES 
表示 大 范围 的 htmlentities0 函数 所 用 到 的 转换 内 容 ，HTML SPECIALCHARS 表示 小 范围 的 
htmlspecialchars0 函 数 所 用 到 的 转换 内 容 ， 第 二 个 参数 的 三 种 模式 ENT COMPAT、ENT QUOTES、 
ENT_NOQUOTES 的 作用 ， 可 以 查看 htmlentities0 函 数 中 的 参数 。 该 函数 的 使 用 示例 如 下 所 示 : 

<?php 

print rl(get html translation table (HTML SPECIALCHARS,ENT QUOTES)); 

echo "<hr>"; 


print rl(get html translation table (HTML ENTITIES,ENT QUOTES)); 
人 


对 于 该 示例 ， 读 者 最 好 亲自 动手 做 一 下 ， 然 后 查看 源 文件 ， 会 产生 一 个 比较 有 意思 的 结果 。 

(5) strtr0 

该 函数 第 一 种 方式 使 用 三 个 参数 ， 蔡 换 时 会 将 第 二 个 参数 的 字符 串 蔡 换 为 第 三 个 参数 的 字符 串 
〈 如 果 两 个 字符 串 参数 的 长 度 不 一 致 , 则 较 长 的 那个 字符 串 将 会 被 截断 ), 第 二 种 方式 使 用 两 个 参数 ， 
第 二 个 参数 是 一 个 准备 进行 蔡 换 处 理 的 数组 。 该 函数 的 使 用 示例 如 下 所 示 : 

<?php 

Strans = strtr("hello world", "world", "body"); 1 

echo $trans;// 显 示 heyyo bodyd (w=>b, 0=>o, r=>d,， 1=>y) 

echo "<br>"; 

$arr = array ("hello" => "hi", "hi" => "hello"); 

Strans = Strtr(l"hi all; 1 satd hellor, Sa 2 


echo $trans;// 显 示 hello all, I said hi 
2 


在 上 述 代 码 的 第 1 行 中 ， 第 二 个 参数 和 第 三 个 参数 指定 葵 换 模板 ， 这 里 会 把 字符 “w” 葵 换 为 
“b”， 字符“o” 蔡 换 为 “o”， 字 符 “r” 蔡 换 为 “d”， 以 此 类 推 ， 模 板 获得 后 ， 就 可 以 对 第 一 个 参数 
中 的 字符 进行 替换。 在 第 2 行 中 ， 字 符 的 蔡 换 模板 是 在 数组 中 指定 的 。 

2. 将 HTML 转换 为 纯 文本 

把 HTML 标记 的 信息 转换 为 纯 文 本 信息 ， 有 时 是 非常 有 用 的 。 下 面 这 个 函数 就 可 以 完成 这 项 
功能 。 

strip_tagsO 函 数 的 主要 功能 是 去 掉 HIML 及 PHP 的 标记 。 该 函数 的 语法 格式 如 下 所 示 : 


string strip tags ( string $str [, string $allowable tags] ) 


第 二 个 参数 表示 人 允许 出 现 的 标签 对 。 若 字符 串 的 HTML 及 PHP 标签 原来 就 有 错 , 例如 少 了 “>” 
符号 ， 则 也 会 返回 错误 。 而 该 函数 和 fgetss0 函 数 具 有 相同 的 功能 。 该 函数 的 使 用 示例 如 下 所 示 : 


<?php 
Stext = "<p>Test paragraph.</p><em>Other text</em>"'; 


echo strip _tags ($text,'<p>'); // 显示 <p>Test paragraph.</P>Other text 


| 


在 上 述 代 码 中 ，HTML 标记 <p> 可 以 不 被 去 除 。 如 果 查 看 注释 中 的 显示 结果 ， 需 要 查看 该 网 页 


的 源 文件 。 


8.3.5 ”正则 表达 式 函 数 的 替代 函数 


从 前 面 的 章节 可 以 知道 ， 利 用 正则 表达 式 可 以 对 字符 串 信息 进行 匹配 、 蔡 换 、 拆 分 等 。 但 在 一 


般 情况 下 不 使 月 


日 正则 表达 式 ， 因 为 正则 表达 式 的 执行 速度 较 慢 ， 只 有 在 信息 比较 复杂 的 情况 下 才 使 


用 。 在 处 理 比较 简单 的 信息 时 ， 通 常 使 用 下 列 函 数 ， 其 详细 信息 如 表 8-15 所 示 。 


名 称 
strtokO 
parse str0 
explode0 


implodeO 
strposO 


stripos() 
strrops() 
str_replace(O) 
str ireplace(O) 
strstr() 
substr() 


substr_countO 


substr_replace 


String strtok (string $str, string $token) 

Void parse_str (string $str [, array &$arr]) 

array explode (string $separator, string $string [, 
int $limit]) 

string implode (string $glue, array $pieces) 

int strpos (string $haystack, mixed S$needle [, int 
S$offset]) 

int stripos (string $haystack, string $needle [, int 
S$offset]) 

int strrpos (string $haystack, mixed $needle [. int 
S$offset]) 

mixed str_replace (mixed $search, mixed S$replace. 
mixed $subject [, int &$count]) 

同上 

String strstr (string $haystack. string $needle) 
string substr (string $string, int $start [, int 
Slength]) 

int substr_count (string $haystack. string $needle 
[. int $offset [. int $length]]) 

mixed substr replace (mixed $string. 
S$replacement. int $start [. int $length]) 


string 


1. strtok() 

该 函数 将 字符 串 str 依 字 符 串 token 的 值 切 开 分 成 小 段 的 字符 串 。 此 函数 的 功能 较 强 大 , 但 是 使 
用 方法 比较 特殊 。 在 连续 使 用 时 ， 第 一 次 需要 说 明 字 符 串 和 分 割 符 ， 但 是 第 二 次 ， 只 要 放 入 分 割 符 
就 可 以 了 ， 分 割 符 可 以 是 多 个 内 容 ， 但 是 不 能 使 用 字符 串 。 该 函数 的 使 用 示例 如 下 所 示 : 


<?php 


$string = "This is\tan example\nstring"; 
So = strtok(Sstringe " NVE")S 
while ($tok !== false) { 

echo "Stok<br />"; 


表 8-15 ”正则 表达 式 蔡 代 函数 


语法 格式 


功 能 
用 指定 的 若干 个 字符 来 分 割 字符 串 
把 字符 串 解析 成 变量 


使 用 一 个 字符 串 分 割 另 一 个 字符 串 


将 数组 的 内 容 组 合成 一 个 字符 串 
寻找 字符 串 中 某 字 符 最 先 出 现 处 


返回 字符 串 中 要 匹配 的 字符 最 先 出 现 的 位 置 
寻找 字符 串 中 某 字 符 最 后 出 现 处 

可 以 匹配 和 替换 字符 串 

可 以 匹配 和 替换 字符 串 ， 忽 略 大 小 写 

通过 比较 返回 一 个 字符 串 的 部 分 
获取 部 分 字符 串 

获取 某 个 字符 在 字符 串 中 出 现 的 次 数 


蔡 换 字符 串 中 部 分 内 容 
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在 上 述 代码 中 ， 其 分 隔 符 分 别 为 空格 、、 等 。 

2. parse_str() 

该 函数 的 作用 是 把 一 定格 式 的 字符 串 转变 为 变量 和 值 ， 并 把 变量 设置 在 当前 作用 域内 ， 字 符 串 
的 格式 和 URL 的 格式 相同 。 在 处 理 URL 时 ,如 果 其 中 包含 HTML 表单 或 者 其 他 通过 查询 字符 串 传 
递 过 来 的 参数 ， 该 函数 就 特别 有 用 。 第 二 个 参数 是 可 选 的 ， 表 示 把 转换 后 的 值 存储 到 数组 中 。 该 函 
数 的 使 用 示例 如 下 所 示 : 


这 里 需要 注意 的 是 ， 字 符 串 的 开始 位 置 如 果 是 一 个 字符 “? ”， 将 无 法 解析 出 该 字符 串 的 第 一 
个 变量 和 值 。 

3. explode() 

该 函数 返回 由 字符 串 组 成 的 数组 ， 每 个 元 素 都 是 string 的 一 个 子 串 ， 它 们 被 字符 串 separator 作 
为 边界 点 分 割 出 来 。 如 果 设 置 了 limit 参数 ， 则 返回 的 数组 包含 最 多 limit 个 元 素 ， 而 最 后 那个 元 素 
将 包含 string 的 剩余 部 分 。 如 果 separator 为 空 字符 串 〈"")， 则 返回 False。 如 果 separator 所 包含 的 
值 在 string 中 找 不 到 ， 则 返回 包含 string 单个 元 素 的 数组 。 如 果 limit 参数 是 负数 ， 则 返回 除了 最 后 
的 -limit 个 元 素 外 的 所 有 元 素 。 该 函数 的 使 用 示例 如 下 所 示 : 


4. implode() 
该 函数 的 参数 glue 是 字符 之 间 的 分 隔 符号 。 该 函数 的 使 用 示例 如 下 所 示 : 


5. strpos() 
该 函数 查找 并 返回 首 个 匹配 项 的 位 置 , 第 一 个 参数 表示 处 理 的 字符 串 , 第 二 个 参数 表示 匹配 项 ， 
第 三 个 参数 可 选 ， 表 示 开始 执行 查找 的 位 置 。 区 分 大 小 写 。 该 函数 的 使 用 示例 如 下 所 示 : 


在 上 述 代 码 中 ， 需 要 注意 的 是 函数 开始 查找 的 位 置 。 
6. stripos() 
该 函数 的 功能 和 strpos0 函 数 类 似 , 区 别 是 匹配 时 不 会 区 分 大 小 写 。 该 函数 的 使 用 示例 如 下 所 示 : 


7. strrops() 

该 函数 用 来 寻找 字符 串 haystack 的 字符 needle 最 后 出 现 的 位 置 .值得 注意 的 是 needle 只 能 是 一 
个 字符 ， 不 能 是 中 文字 符 等 。 若 找 不 到 指定 的 字符 ， 则 返回 False 值 。 最 后 一 个 参数 是 可 选 的 ， 表 
示 开 始 查 找 的 位 置 。 


8. str_replace() 
该 函数 的 第 一 个 参数 表示 需要 匹配 的 项 ， 第 二 个 参数 表示 蔡 换 的 项 ， 第 三 个 参数 表示 处 理 的 字 
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符 串 ， 第 四 个 参数 可 选 ， 表 示 将 一 个 变量 赋值 为 匹配 的 次 数 ， 前 两 个 参数 也 可 以 是 数组 形式 的 。 该 
函数 的 使 用 示例 如 下 所 示 : 


9. str_ireplace() 
该 函数 的 功能 和 str_replaceO 函 数 类 似 ， 只 不 过 在 匹配 时 str_ireplace0 函 数 会 忽略 大 小 写 。 该 函 
数 的 使 用 示例 如 下 所 示 : 


10. strstr() 
该 函数 有 两 个 参数 ， 第 一 个 参数 表示 处 理 的 字符 串 ， 第 二 个 参数 表示 匹配 项 ， 返 回 结果 为 从 匹 
配 位 置 开始 到 最 后 的 一 个 字符 串 ， 如 果 没 有 匹配 ， 则 返回 False。 该 函数 的 使 用 示例 如 下 所 示 


11. substr() 

该 函数 可 以 对 字符 串 进行 截取 ， 它 有 三 个 参数 ， 第 一 个 参数 表示 要 操作 的 对 象 。 第 二 个 参数 如 
果 是 空 ， 那 么 默认 为 0， 如 果 是 负数 ， 就 表示 从 后 往 前 计数 ， 第 三 个 参数 如 果 是 正 数 ， 表 示 字 符 串 
截取 的 长 度 ， 如 果 省 略 ， 则 会 一 直 截 取 到 最 后 ， 如 果 是 负数 ， 则 表示 从 后 往 前 截取 到 的 位 置 。 该 函 
数 的 使 用 示例 如 下 所 示 : 


12. substr_count() 

该 函数 的 作用 是 计算 字符 串 中 某 字 符 出 现 的 次 数 ， 第 一 个 参数 表示 在 该 字符 串 中 查找 ， 第 二 个 
参数 表示 要 查找 的 字符 串 ， 第 三 个 参数 表示 在 字符 串 中 开始 比较 的 位 置 ， 省 略 则 默认 从 头 开始 ， 第 
四 个 参数 表示 依次 比较 的 字符 数 ， 省 略 则 表示 一 直 比 较 到 末尾 。 和 substr0 函 数 有 所 不 同 的 是 ,这 两 
个 参数 都 不 支持 负数 。 该 函数 的 使 用 示例 如 下 所 示 : 


<?php 

$text = "This is a test"7 

echo substr count (Stext， "is'); // 显示 2 

echo substr count ($text，'is'，3); // 显示 1 
echo substr count (Stext， "is'，3，3)7 // 显示 0 
2 


13. substr_replace 

该 函数 的 第 一 个 参数 表示 在 该 字符 串 中 查找 并 蔡 换 ， 第 二 个 参数 表示 葵 换 的 内 容 ， 第 三 个 参数 
必 选 ， 表 示 需 要 处 理 字符 的 起 始 位 置 ， 负 数 表 示 从 后 往 前 计数 ， 第 四 个 参数 可 选 ， 省 略 表示 一 直 蔡 
换 到 最 后 ， 正 数 表 示 蔡 换 的 长 度 ， 负 数 表示 从 后 往 前 替换 到 的 位 置 。 该 函数 的 使 用 示例 如 下 所 示 : 

<?php 

$var = 'ABCDEFGH:/MNRPQR/'; 

echo substr replace ($var，'bob'，0); // 显示 bob 

echo substr replace($var，'bob'，0，0); // 显示 bobABCDEFGH:/MNRPQR/ 

echo substr replace($var，'bob'，10，-1); // 显示 ABCDEFGH:/bob/ 

echo substr _ replace ($var，'bob'，-7，-1); // 显示 ABCDEFGH:/bob/ 

echo substr replace($var,''，10，-1); // 显示 ABCDEFGH:// 

?> 


8.3.6 ”填充 和 剔除 字符 串 


有 时 为 了 Web 程序 的 需要 ,需要 把 字符 串 中 的 一 个 字符 去 掉 , 或 者 在 字符 串 中 添加 一 些 新 的 字 
符 。 为 实现 该 功能 ，PHP 提供 了 一 些 函 数 。 本 节 将 一 一 介绍 这 些 函 数 。 其 详细 信息 如 表 8-16 所 示 。 
表 8-16 ”填充 和 剔除 函数 


名 称 | 语法 格式 | 功 能 


lnim0 去 除 左边 连续 空白 和 指定 字符 
rtrim0) string rtrim (string $str [. string $charlist]) 去 除 右边 连续 空白 和 指定 字符 
trim0 | string trim (string $str [. string $charlist]) 删除 字符 串 中 指定 的 字符 
str_pad() string str pad (string $input int $pad length [. string $pad | 在 指定 的 字符 串 中 添加 字符 


string [. int $pad_type]]) 


1. ltrim() 

该 函数 去 除 字符 串 左 侧 的 空白 或 者 指定 的 字符 , 这 些 指 定 的 字符 包括 空格 符 、 水 平 制 表 符 (\t)、 
换行 、 回 车 (\r)、 空 值 \0)、 垂 直 制 表 符 (x0b)。 如 果 去 掉 其 他 的 字符 ， 如 字符 串 中 的 a 字符 ， 
可 以 使 用 第 二 个 参数 。 该 函数 的 使 用 示例 如 下 所 示 : 


<?php 
$text = "\t\tThese are a few words :) ... ™; 
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2. rtrim() 
该 函数 和 ltrimO) 函 数 的 功能 相同 ， 只 是 它 从 str 的 右 侧 删除 字符 。 该 函数 的 使 用 示例 如 下 所 示 : 


3. trim() 
该 函数 是 ltrim0 和 rtrim0 函 数 的 组 合 ， 该 函数 可 以 从 字符 串 的 两 侧 删 除 字符 串 中 的 指定 字符 。 
该 函数 的 使 用 示例 如 下 所 示 : 


4. str_pad() 

该 函数 的 作用 是 对 字符 串 进行 两 侧 的 补 白 ， 第 一 个 参数 表示 被 操作 的 字符 串 对 象 。 第 二 个 参数 
表示 补 白 以 后 的 长 度 ， 第 三 个 参数 可 选 ， 表 示 进 行 补 白 的 字符 串 ， 默 认为 空格 ， 第 四 个 参数 可 选 ， 
表示 补 白 的 方式 ， 有 三 个 常量 可 以 选择 : STR_PAD RIGHT、STR_PAD LEFT 和 STR _PAD BOTH， 
分 别 表示 右 补 白 、 左 补 白 和 两 侧 补 白 ， 默 认 值 为 STR_PAD _ RIGHT。 该 函数 的 使 用 示例 如 下 所 示 : 


8.3.7 ”字符 和 单词 计数 


在 PHP 程序 中 ， 获 取 一 个 字符 串 中 单词 或 字符 的 总 数 有 时 是 必需 的 ， 实 现 这 些 功能 非常 简单 ， 
只 需要 调用 相应 的 PHP 函数 。 本 小 节 将 会 介绍 这 两 个 函数 。 其 详细 信息 如 表 8-17 所 示 。 


表 8-17 ”字符 和 单词 计数 函数 
名 称 语法 格式 格 式 
count chars() mixed count chars (string $string [. int $mode]) 返回 字符 串 所 用 字符 的 信息 


str_ word_count() | mixed str word count (string $string [， int $format [. | 获取 字符 串 中 的 单词 总 数 的 信息 8 
string $charlist]]) i 


1. count_chars() 

该 函数 主要 用 来 统计 string 中 每 个 字 节 值 (0~255〉 出 现 的 次 数 ， 使 用 多 种 模式 返回 结果 。 可 
选 参数 mode 默认 值 为 0。 

根据 不 同 的 mode，count_chars0 函 数 返回 下 列 不 同 的 结果 : 


。 0 以 所 有 的 每 个 字 节 值 作为 键 名 ， 出 现 次 数 作为 值 的 数组 。 第 
e。 1 与 0 相同 ， 但 只 列 出 出 现 次 数 大 于 零 的 字 节 值 。 8 
。 2 与 0 相同 ， 但 只 列 出 出 现 次 数 等 于 零 的 字 节 值 。 sa 
。 3 返回 由 所 有 使 用 了 的 字 节 值 组 成 的 字符 事 。 二 
。4 返回 由 所 有 未 使 用 的 字 节 值 组 成 的 字符 囊 。 中 
该 函数 的 使 用 示例 如 下 所 示 ; 加 
<?php 则 


$data = "this is a test world"; 

print r(count chars ($data，1)); // 显示 用 到 字符 信息 
print r(count chars($data，3)); // 显示 adehilorstw 
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2. str_word_count() 

该 函数 的 作用 是 获取 字符 串 中 的 英文 单词 信息 ， 第 一 个 参数 表示 要 操作 的 字符 串 对 象 ， 第 二 个 
参数 可 选 ，0 表示 只 返回 单词 的 个 数 ，1 表示 将 找到 的 单词 作为 一 个 数组 返回 ，2 表示 将 找到 的 单词 
和 单词 所 在 的 字符 位 置信 息 作为 一 个 联合 数组 (associative) 返回 ， 默 认 值 是 0， 第 三 个 参数 也 是 可 
选 的 ， 表 示 需 要 忽略 的 断 词 符号 ， 默 认 的 断 词 符号 有 空格 、 数 字 等 ， 有 时 需要 忽略 一 些 。 该 函数 的 
使 用 示例 如 下 所 示 : 


<?php 
$str = "Hello fri3nd, you're 
looking good today!™"; 
echo str word count ($str); // 显示 7 
print r(str word count ($str, 1));// 显示 Array ( [0] => Hello [1] => fri [2] => nd 
[3] => good [4] => today ) 
print r(str word count ($str, 2)); // 显示 Array ( [0] => Hello [6] => fri [10] => 
nd [14] => good [19] => today ) 
print r(str word count ($str, 1，'"'aé~ac3')); // 显示 Array ( [0] => Hello [1] => 
fri3nd [2] => good [3] => today ) 
2 
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PHP 语言 提供 了 大 量 工具 ， 不 仅 可 以 处 理 文件 系统 的 输入 和 输出 ， 还 可 以 在 Shell 级 执行 程序 。 
其 中 文件 是 由 相关 数据 组 成 的 实体 ; 文件 的 所 有 权 和 权限 是 所 有 主流 操作 系统 提供 的 一 种 保护 系统 
数据 的 方式 ， 也 就 是 通过 基于 用 户 和 用 户 组 所 有 权 及 权限 的 权限 系统 来 保护 系统 数据 ; Shell 命令 是 
用 户 通过 一 些 内 置 函数 和 机 制 ， 可 以 在 PHP 应 用 程序 中 利用 操作 系统 和 其 他 语言 级 的 功能 。 


和 内 容 摘要 lncwsct _ 


了 解 文件 和 目录 

掌握 如 何 解析 目录 路 径 

掌握 如 何 获得 文件 类 型 和 连接 
掌握 如 何 计 算 文件 、 目 录 和 磁盘 大 小 
掌握 如 何 访问 和 修改 时 间 
理解 文件 所 有 权 和 权限 

掌握 文件 IO 的 相关 操作 

掌握 如 何 执行 Shell 命令 
理解 系统 程序 的 执行 


9.1 了 解 文 件 和 目录 


文件 是 存储 在 硬盘 驱动 器 、CD-ROM 或 其 他 媒体 上 的 一 系列 字 节 。 文 件 具 有 名 称 ， 这 样 可 以 方 
便 地 引用 它们 。 除 此 之 外 ， 文 件 还 具有 位 置 、 大 小 、 最 后 修改 时 间 、 最 后 访问 时 间 及 其 他 文件 的 信 
息 。 为 了 更 容易 地 处 理 文件 ， 可 以 把 文件 组 织 在 目录 中 。 并 且 一 个 目录 可 以 包含 其 他 目录 ， 这 些 目 
录 称 为 它 的 子 目录 ， 而 它 称 为 这 些 目录 的 父 目 录 。 多 个 父 目 录 和 子 目 录 形 成 了 一 个 单独 的 目录 树 或 
分 层 结构 。 目 录 树 中 最 上 层 的 目录 称 为 根 目录 ， 记 作 /， 其 他 所 有 目录 都 是 根 目录 的 子 目录 。 
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9.1.1 解析 目录 路 径 


PHP 提供 了 一 些 有 助 于 处 理 路 径 的 函数 ， 如 表 9-1 所 示 。 通 过 这 些 函数 可 以 解析 目录 路 径 ， 如 
获得 末尾 的 扩展 名 、 目 录 部 分 和 基本 名 等 。 


表 9-1 解析 目录 路 径 函 数 


函数 说 明 


basenameO | 该 函数 用 于 返回 路 径 中 的 文件 名 部 分 。 它 给 出 一 个 包含 有 指向 一 个 文件 的 全 路 径 的 字符 串 ， 该 函数 
返回 基本 的 文件 名 。 如 果 文 件 名 是 以 suffix 结束 的 ， 那 么 这 一 部 分 也 会 被 去 掉 。 在 Windows 中 ， 斜 
线 (/) 和 反 斜 线 〈\) 都 可 以 用 作 路 径 分 隔 符 。 在 其 他 环境 下 是 斜 线 (/) (比如 Linux、UNIX 等 ) 


dimame0 | 该 函数 用 于 返回 路 径 中 的 目录 部 分 。 它 给 出 一 个 包含 有 指向 一 个 文件 的 全 路 径 的 字符 串 ， 该 函数 返 
回去 掉 文件 名 后 的 目录 名 


pathinfo0 ”| 该 函数 用 于 返回 文件 路 径 的 信息 。 它 返回 一 个 结合 数组 包含 有 path 的 信息 ， 包 括 以 下 的 数组 单元 : 
dirmame (目录 名 )、basename (基本 名 ) 和 extension (扩展 名 ) 


realpathO ”| 该 函数 用 于 返回 规范 化 的 绝对 路 径 名 。 它 扩展 所 有 的 符号 连接 , 并 且 处 理 输入 的 path 中 的 /.\Y../ 及 
多 余 的 /， 然 后 返回 规范 化 后 的 绝对 路 径 名 。 返 回 的 路 径 中 没有 符号 连接 ，V./ 或 /./ 成 分 。realpathO 
失败 时 则 返回 False， 例 如 文件 不 存在 时 


下 面 分 别 对 表 9-1 所 示 的 函数 的 格式 进行 介绍 及 举例 。 
1. basename() 
它 的 使 用 格式 如 下 所 示 : 


string basename ( string path [, string suffix]); 
使 用 basename0 函 数 的 具体 示例 如 下 所 示 : 


<?php 

$path = "/9/jack/test.txt"; 

// 取 得 filename 为 : test.txt 

$filename = basename ($path); 

// 取 得 filename2 为 : test 

$filename?2 = basename ($path,".txt"); 
?> 
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2. dirname() 
它 的 使 用 格式 如 下 所 示 : 


string dirname ( string path); 
使 用 dimameO 函 数 的 具体 示例 如 下 所 示 : 


<?php 

$path = "/9/jack/test.txt"; 
// 取得 $file 为 : /9/jack 
$file = dirname ($path); 
> 

3. pathinfo() 

它 的 使 用 格式 如 下 所 示 : 


array pathinfo ( string path); 
使 用 pathinfo0 函 数 的 具体 示例 如 下 所 示 : 


<?php 
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$path parts = pathinfo("/www/htdocs/index.php"); 
echo $path parts["dirname"] . "\n"7 

echo $path parts["basename"] . "\n™; 

echo $path parts["extension"] . "\n"; 

> 


上 述 代 码 成 功 执行 后 ， 将 输出 类 似 于 如 下 信息 : 


/www/htdocs 
index.php 
php 


4. realpath() 
它 的 使 用 格式 如 下 所 示 : 


string realpath ( string path); 


使 用 realpath0 函 数 的 具体 示例 如 下 所 示 : 


册 吧 站 
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<?php 

$filepath = "../../index.php"; 

// 获 得 Sreal_path 为 : /www/htdocs/9/index.php 
$real path = realpath ($filepath); 

Ry 
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PHP 提供 了 一 些 函数 来 允许 用 户 获得 文件 系统 中 文件 类 型 和 连接 的 相关 信息 。 下 面 对 这 些 函 数 
进行 介绍 。 

1. filetype() 

它 的 使 用 格式 如 下 所 示 : 


string filetype ( string filename) 


该 函数 用 于 取得 文件 类 型 。 它 返回 文件 的 类 型 可 能 值 有 8 种 ， 如 表 9-2 所 示 。 如 果 出 错 则 返回 
False。 如 果 statO 函 数 调用 失败 或 者 文件 类 型 未 知 ， 还 会 产生 一 个 E_ NOTICE 消息 。 


表 9-2 文件 类 型 
文件 类 型 说 明 
block 块 设备 ， 如 CD-ROM 或 软件 驱动 器 
char 字符 设备 ， 负 责 在 操作 系统 和 设备 〈 如 打印 机 ) 之 间 进 行 无 缓冲 的 数据 交换 
dir 目录 
fifo 命名 管道 ， 常 用 于 将 信息 从 一 个 进程 传递 到 另 一 个 进程 
file 硬 连接 ， 作 为 文件 inode 的 指针 ， 只 要 认为 是 一 个 文件 ， 如 文本 文档 或 可 执行 文件 ， 都 会 返回 
这 个 类 型 
link 符号 连接 ， 是 指向 文件 指针 的 指针 
Socket 套 接 字 资 源 
Unknown 未 知 类 型 


181 


卫 
= 
五 
十 
三 
< 
op) 
oO 
| 
完 
全 
学 
习 
手 
册 


这 里 需要 注意 的 是 ， 该 函数 的 结果 会 被 缓存 ， 并 且 不 能 作用 于 远程 文件 ， 被 检查 的 文件 必须 通 
过 服务 器 的 文件 系统 访问 。 甸 etype0 函 数 的 具体 示例 类 似 于 如 下 所 示 : 


<?php 

// 以 下 语句 输出 : file 

echo filetype('jack/test.txt"'); 
// 以 下 语句 输出 : dir 

echo filetype('jack/'); 

2 


2. link() 
它 的 使 用 格式 如 下 所 示 : 


int link ( string target, string link); 


该 函数 用 于 创建 一 个 指向 target 的 硬 连接 link。 如 果 成 功 则 返回 Tue， 和 否则 返回 False。 因 为 只 
有 作为 服务 器 守护 进程 的 所 有 者 来 执行 PHP 脚本 ， 所 以 如 果 用 户 没有 对 link 所 在 目录 的 写 入 权限 ， 
则 此 函数 将 失败 。 

3. linkinfo() 

它 的 使 用 格式 如 下 所 示 : 


int linkinfo ( string path); 


该 函数 用 来 获得 关于 一 个 符号 连接 的 有 用 信息 ， 包 括 大 小 、 最 后 修改 时 间 和 所 有 者 用 户 ID 等 
信息 。 它 返回 lstat0 函 数 提供 的 某 一 项 ， 用 来 验证 一 个 连接 (由 path 指向 的 ) 是 否 确实 存在 。 如 果 
出 错 则 返回 0 或 False。 此 函数 不 能 用 于 Windows 平台 。 

4. readlink() 

它 的 使 用 格式 如 下 所 示 : 


string readlink ( string path); 


该 函数 用 于 返回 符号 连接 指向 的 目标 ， 如 果 发 生 错 误 则 返回 False， 否 则 返回 符号 连接 的 内 容 。 
假如 ，img_link.txt 是 一 个 指向 img.txt 的 符号 连接 ， 可 以 用 下 面 的 代码 来 获得 img.txt 的 绝对 路 径 : 


Srpath = readlink("/9/jack/img-link.txt"); 
echo "$rpath"; // 输 出 : /9/jack/img.txt 


5. symlink() 
它 的 使 用 格式 如 下 所 示 : 
int symlink ( string target, string link); 


该 函数 用 于 建立 一 个 指向 已 经 存在 的 target 的 符号 连接 , 它 对 于 已 有 的 target 建立 一 个 名 为 link 
的 符号 连接 。 如 果 成 功 则 返回 True， 和 否则 返回 False。 需 要 注意 的 是 该 函数 并 未 在 Windows 平台 下 
实现 。 

6. stat() 

它 的 使 用 格式 如 下 所 示 : 


array stat ( String filename) 


该 函数 用 于 获取 由 锯 ename 指定 的 文件 的 统计 信息 。 如 果 外 ename 是 符号 连接 ， 则 统计 信息 是 
关于 被 连接 文件 本 身 ， 而 不 是 符号 连接 。lstat0 函 数 与 stat0 函 数 基 本 相同 ,不同 的 是 lstat0 函 数 能 返 
回 符号 连接 的 状态 。 

该 函数 返回 一 个 数组 包含 有 文件 的 统计 信息 ， 该 数组 具有 的 单元 如 表 9-3 所 示 ， 数 组 下 标 从 零 
开始 。 除 了 数字 索引 之 外 自 PHP 4.0.6 起 还 可 以 通过 关联 索引 来 访问 。 

表 9-3 ”返回 数组 具有 的 单元 


数组 下 标 | 关联 键 名 说 有明 

0 dev device number 文件 所 在 的 设备 号 

1 ino inode number 文件 的 inode 号 

2 mode inode protection mode 文件 inode 保护 模式 ， 这 个 值 确定 指派 给 文件 的 访问 和 
修改 权限 

3 nlink number of links 与 该 文件 关联 的 硬 连 接 数 目 

4 uid userid of owner 文件 所 有 者 的 用 户 ID (UID) 

5 gid groupid of owner 文件 所 有 者 的 组 ID (GID) 

6 rdev device type,if inode device ”设备 类 型 ， 如 果 是 inode 设备 的 话 。 注 意 Windows 
平台 不 可 用 

学 size size in bytes ”文件 大 小 ， 以 字 节 为 单位 

8 atime time of last access (unix timestamp) 文件 的 最 后 访问 时 间 (UNIX 时 间 玲 格式 ) 

9 mtime time of last modification (unix timestamp) 最 后 修改 时 间 (UNIX 时 间 惟 格式 ) 

10 ctime time oflast change (unix timestamp) ”最 后 改变 时 间 (UNIX 时 间 稚 格 式 ) 

11 blksize blocksize of filesystem ”文件 系统 的 块 大 小 ， 注 意 Windows 平台 不 可 用 

12 blocks number of blocks allocated ”所 占据 块 的 数目 


在 表 9-3 中 所 提 及 的 inode 号 是 与 每 个 文件 名 关联 的 唯一 数值 标识 符 , 用 来 引用 inode 表 中 的 关 
联 项 ， 其 中 包含 有 关 文 件 的 大 小 、 类 型 、 位 置 和 其 他 关键 特性 的 信息 。 

这 里 需要 注意 的 是 stat0 函 数 的 结果 会 被 缓存 ， 并且 不 能 作用 于 远程 文件 , 被 检查 的 文件 必须 通 
过 服务 器 的 文件 系统 访问 。 

7. lstat() 

它 的 使 用 格式 如 下 所 示 : 

array lstat ( string filename) 

该 函数 用 于 给 出 一 个 文件 或 符号 连接 的 信息 。 它 获取 由 filename 指定 的 文件 或 符号 连接 的 统计 
信息 。 该 函数 和 stat0 函 数 相同 ， 当 filename 参数 不 是 符号 连接 时 ， 该 符号 连接 的 状态 被 返回 ， 而 不 
是 该 符号 连接 所 指向 的 文件 状态 。 这 里 需要 注意 的 是 ， 该 函数 的 结果 会 被 缓存 ， 并 且 不 能 作用 于 远 
程 文件 ， 被 检查 的 文件 必须 通过 服务 器 的 文件 系统 访问 。 

8. fstat() 

它 的 使 用 格式 如 下 所 示 : 


array fstat ( resource handle) 


该 函数 用 于 通过 已 打开 的 文件 指针 取得 文件 信息 。 它 获取 由 文件 指针 handle 所 打开 文件 的 统计 
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信息 。 该 函数 和 statO 函 数 相似 ， 不 同 的 是 它 作用 于 已 打开 的 文件 指针 而 不 是 文件 名 。 该 函数 返回 一 
个 具有 该 文件 的 统计 信息 的 数组 。fstatO 函 数 的 具体 示例 如 下 所 示 : 


上 述 代码 成 功 执行 后 ， 将 输出 类 似 于 如 下 信息 : 


9.1.3 ”计算 文件 、 目 录 和 磁盘 大 小 


在 程序 开发 中 常常 需要 计算 某 一 或 某 些 文件 、 目 录 及 磁盘 的 大 小 , 为 此 PHP 提供 了 一 些 函 数 来 
实现 这 些 功 能 ， 通 过 使 用 这 些 函 数 或 函数 的 组 合 可 以 很 容易 地 计算 出 文件 、 目 录 及 磁盘 的 大 小 。 
1. filesize() 


它 的 使 用 格式 如 下 所 示 : 


该 函数 用 于 返回 文件 大 小 的 字 节 数 ， 如 果 出 错 则 返回 False。 因 为 PHP 的 整 型 是 有 符号 的 ， 并 


且 大 多 数 平台 使 用 32 位 整数 , filesize0 函 数 在 碰 到 大 于 2GB 的 文件 时 可 能 会 返回 非 预期 的 结果 , 所 
以 对 于 2GB~4GB 之 间 的 文件 通常 可 以 使 用 sprintf("%u",filesize($file)) 来 解决 此 问题 。 这 里 需要 注意 
的 是 ， 该 函数 的 结果 会 被 缓存 ， 并 且 不 能 作用 于 远程 文件 ， 被 检查 的 文件 必须 通过 服务 器 的 文件 系 
统 访问 。fiesize0 函 数 的 具体 示例 如 下 所 示 : 

<?php 

// 输出 类 似 于 jack/itying.txt: 463 bytes 

$filename = "jack/itying.txt'7 

echo $filename . ': ' . filesize($filename) . ' bytes'; 

> 


上 述 代码 成 功 执行 后 ， 将 输出 类 似 于 如 下 信息 : 


jack/itying.txt: 463 bytes 


2. disk_free_space() 
它 的 使 用 格式 如 下 所 示 : 


float disk free space ( string directory); 
该 函数 用 于 返回 目录 中 的 可 用 空间 。 上 述 格式 给 出 一 个 包含 有 一 个 目录 的 字符 串 ， 该 函数 将 根 
据 相 应 的 文件 系统 或 磁盘 分 区 返回 可 用 的 字 节 数 。disk_free_space0 函 数 的 具体 示例 如 下 所 示 : 


<?php 

// 返回 根 目录 下 可 用 的 字 节 数 

$dfs = disk free space("/"); 
echo "dfs="'.$dfs.'bytes'; 

?> 


上 述 代 码 成 功 执行 后 ， 将 输出 类 似 于 如 下 信息 : 


dfs=6235213824bytes 


3. disk_total_space() 
它 的 使 用 格式 如 下 所 示 : 


float disk total space ( string directory); 


该 函数 用 于 返回 一 个 目录 的 磁盘 总 大 小 。 确切 地 说 , 返回 的 是 该 目录 所 在 的 磁盘 分 区 的 总 大 小 ， 
因此 将 同一 个 磁盘 分 区 的 不 同 目录 作为 参数 所 得 到 的 结果 完全 相同 。 在 UNIX 和 Windows NT 中 都 
支持 将 一 个 磁盘 分 区 加 载 为 一 个 子 目 录 ， 这 时 该 函数 与 disk_free_space0 函 数 结合 就 很 有 用 。 
disk total space0 函 数 的 具体 示例 如 下 所 示 : 

<?php 

// 返 回 目录 所 在 磁盘 的 总 大 小 

$df = disk total space("/") 7 

$dfs = round ($df/1048576,2); 

echo "dfs="*.$dfs. "MB'; 

?> 
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上 述 代码 成 功 执行 后 ， 将 输出 类 似 于 如 下 信息 : 
[RCR 


这 里 需要 注意 的 是 ， 该 返回 值 的 数量 单位 是 兆 字 节 (MB)， 因 为 在 上 面 将 disk total spaceO 函 
数 的 返回 值 除 以 1048576 字 节 (1MB)。 

4. 获取 目录 大 小 

目录 PHP 虽然 提供 了 filesize0、disk free space0 及 disk total space0 函 数 ， 但 并 没有 提供 获取 
目录 总 大 小 的 标准 函数 。 也 许 会 有 读者 认为 ， 可 以 使 用 exec0 或 system0 做 系统 级 调用 du 命令 (用 
于 获得 一 个 文件 或 目录 的 磁盘 使 用 情况 ), 但 由 于 存在 安全 方面 的 问题 , 这些 函数 通常 是 禁用 的 。 为 
此 不 得 不 采用 另 一 种 解决 方法 ， 该 方法 通过 一 个 自 定义 的 PHP 函数 来 完成 。 具 体 代码 如 下 所 示 : 


将 上 述 代 码 存储 在 9-2.php 文件 中 ， 保 存 到 C:\Apache2.2\htdocs\9 中 ， 然 后 打开 正 浏览 器 ， 在 


地 址 栏 中 输入 http://localhost/9/9-2.php， 运 行 结果 如 下 所 示 : 


Directory jack/: 0.42kb. 


9.1.4 ”访问 和 修改 时 间 


PHP 提供 了 一 些 函 数 用 于 获得 文件 的 上 次 访问 和 修改 的 时 间 ， 这 些 函数 分 别 是 名 eatimeO、 


filectime0 及 filemtime0， 其 详细 信息 如 表 9-4 所 示 。 
表 9-4 ”获取 时 间 函 数 

函数 说 了 明 

fileatime() | 该 函数 用 于 返回 文件 和 锯 ename 上 次 被 访问 的 时 间 , 如 果 出 错 则 返回 False。 时 间 以 UNIX 时 间 戳 的 方 
式 返 回 
该 函数 用 于 返回 文件 flename 上 次 被 修改 的 时 间 , 如 果 出 错 则 返回 False。 时 间 以 UNIX 时 间 稚 的 方 
式 返回 。 这 里 的 修改 时 间 指 的 是 对 文件 inode 数据 的 任何 修改 ， 包 括 改变 权限 、 所 有 者 、 组 或 其 他 
inode 特定 的 信息 
该 函数 用 于 返回 文件 flename 上 次 被 修改 的 时 间 , 如 果 出 错 则 返回 False。 时 间 以 UNIX 时 间 稚 的 方 
式 返回 。 这 里 的 修改 时 间 指 的 是 对 文件 内 容 进 行 修改 的 时 间 


下 面 分 别 对 表 9-4 所 示 的 函数 的 格式 进行 介绍 及 举例 。 
1. fileatime() 
它 的 使 用 格式 如 下 所 示 : 


int fileatime ( string filename); 
使 用 名 eatime0O 函 数 的 具体 示例 如 下 所 示 : 


<?php 

// 输出 文件 的 最 后 访问 时 间 

$filename = 'jack/itying.txt'; 

if (file exists($filename)) { 
echo basename ($filename)." fileaname(): " . date ("F d Y H:i:s.", fileatime 
($filename)); 

» 

a 


上 述 代 码 成 功 执行 后 ， 将 输出 类 似 于 如 下 信息 : 


itying.txt fileaname(): June 21 2007 04:41:07. 


filectime() 


filemtime() 


2. filectime() 

它 的 使 用 格式 如 下 所 示 : 

int filectime ( string filename); 
使 用 flectime0 函 数 的 具体 示例 如 下 所 示 : 


<?php 
// 输出 文件 的 最 后 修改 时 间 
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$filename = 'jack/itying.txt"; 

if (file exists($filename)) { 
echo basename ($filename)." filecname(): " . date ("F d Y H:i:s.", filectime 
($filename)); 

} 

2 


上 述 代码 成 功 执行 后 ， 将 输出 类 似 于 如 下 信息 : 
itying.txt filecname (): June 21 2007 03:21:43. 


3. flemtime() 
它 的 使 用 格式 如 下 所 示 : 


int filemtime ( string filename); 


使 用 锯 emtime0 函 数 的 具体 示例 如 下 所 示 : 


<?php 

// 输出 文件 的 最 后 修改 时 间 

$filename = 'jack/itying.txt'; 

if (file exists($filename)) { 
echo basename ($filename)." filemname():" . date ("F d Y H:i:s.", filemtime 
($filename)); 
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攻关 
上 述 代码 成 功 执行 后 ， 将 输出 类 似 于 如 下 信息 : 


itying.txt filemname(): June 21 2007 06:57:16. 


9.2 文件 所 有 权 和 权限 


文件 所 有 权 和 权限 设置 不 当 ， 将 有 可 能 造成 存储 在 文件 中 的 数据 遭 到 破坏 或 泄露 某 些 机 密 信 
息 ， 给 企业 或 个 人 造成 损失 。 通 常情 况 下 ， 系 统管 理 员 通过 配置 用 户 账号 或 更 改 文件 的 所 有 权 和 权 
限 ， 以 使 文件 免 受 安全 威胁 。 在 PHP 中 也 是 如 此 ， 它 通过 提供 一 些 函 数 来 实现 这 些 操作 ， 并 且 可 以 
获得 文件 的 相关 信息 。 如 表 9-5 所 示 为 与 文件 所 有 权 有 关 的 函数 ;如 表 9-6 所 示 为 文件 具有 的 权限 
函数 。 
表 9-5 文件 所 有 权 函 数 
函数 使 用 格式 说 了 明 
chown() bool chown ( string filename. | 该 函数 尝试 将 文件 filename 的 所 有 者 改 成 用 户 user( 由 用 户 名 或 用 
mixed user): 户 D 指定 )。 只 有 超级 用 户 可 以 改变 文件 的 所 有 者 。 如 果 成 功 则 
返回 Tme， 否 则 返回 False 
chgrmpO bool chgrp ( string filename. | 该 函数 尝试 将 文件 人 lename 所 属 的 组 改 成 group( 通 过 组 名 或 组 ID 
mixed group): 指定 )。 只 有 超级 用 户 可 以 任意 修改 文件 的 组 ， 其 他 用 户 只 能 将 文 


件 的 组 改 成 该 用 户 自己 所 在 的 组 。 如 果 成 功 则 返回 True， 和 否则 返 
回 False 
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说 明 
该 函数 用 于 返回 文件 的 访问 权限 ， 如 果 出 错 则 返回 False 
该 函数 用 于 取得 该 文件 所 属 组 的 ID。 如果 出 错 则 返回 False。 组 ID 
以 数字 格式 返回 ， 可 以 用 posix_getgrgid0 函 数 来 将 其 解析 为 组 名 。 ™ 
如 果 失 败 则 返回 False 以 及 一 个 E_ WARNING 级 别 的 错误 人 
该 函数 用 于 取得 文件 的 所 有 者 。 它 返回 文件 所 有 的 用 户 DD， 如 果 

出 错 则 返回 False。 用 户 ID 以 数字 格式 返回 ， 用 posix_getpwuidO 

函数 来 将 其 解析 为 用 户 名 。 


函数 使 用 格式 
fileperms() | int fileperms ( string filename): 
filegroupO | int filegroup ( string filename): 


fileowner( | int fileowner (string filename):; 


在 表 9-5 中 ，fileperms0 函 数 的 具体 示例 如 下 所 示 : 


<?php 过 

$filename = "jack/itying.txt'7 章 

$filename2 = 'jack/test.txt'; 

echo 'filenamel:'.substr(sprintf('%0o', fileperms ($filename)), -4).'<br>'; 处 

echo 'filename2:'.substr(sprintf('%0', fileperms ($filename2)), -4); 理 

2> 六 
件 

上 述 代码 成 功 执行 后 ， 将 输出 类 似 于 如 下 信息 ; 担 

filenamel:0666 了 

filename2:0666 统 

表 9-6 文件 权限 函数 
函数 说 阴 


is_executable() | 该 函数 用 于 判断 所 给 文件 flename 是 否 可 以 执行 。 如 果 文件 存在 且 可 以 执行 ， 则 返回 True。 该 
函数 从 PHP 5.0.0 起 可 用 于 Windows 系统 

is_readable0 ”| 该 函数 用 于 判断 所 给 文件 flename 是 否 可 读 。 如 果 由 fename 指定 的 文件 或 目录 存在 并 且 可 读 ， 
则 返回 True 

is_writableO0 ”| 该 函数 用 于 判断 所 给 文件 flename 是 否 可 写 。 如 果 文件 存在 并 且 可 写 则 返回 True。 filename 参数 
可 以 是 一 个 允许 进行 是 否 可 写 检查 的 目录 名 。 另 外 要 注意 的 是 ,该 函数 与 is_writeable0 函 数 具 有 


同样 的 功能 
wmaskO 该 函数 用 于 指定 新 创建 文件 的 权限 级 别 。 它 通过 将 mask 与 0777 按 位 与 , 来 计算 PHP 的 umask， 


并 返回 旧 的 掩 码 。mask 是 表示 权限 级 别 的 3 位 或 4 位 数字 代码 。PHP 通过 脚本 创建 文件 和 目录 
时 将 使 用 这 个 umask。 如 果 忽 略 参数 mask， 将 得 到 PHP 当前 配置 的 umask 值 


下 面 分 别 对 表 9-6 所 示 的 函数 的 格式 进行 介绍 及 举例 : 
1. is_executable() 
它 的 使 用 格式 如 下 所 示 : 


bool is executable ( string filename); 
使 用 is_executable0 函 数 的 具体 示例 如 下 所 示 。 


<?php 
$file = "jack/itying.txt'"7 
if (is executable ($file)) { 
echo basename ($file).' is executable'; 
} else { 
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上 述 代码 成 功 执行 后 ， 将 输出 如 下 所 示 的 信息 : 


2. is_readable() 
它 的 使 用 格式 如 下 所 示 : 


使 用 is_readableO 函 数 的 具体 示例 如 下 所 示 : 
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上 述 代码 成 功 执行 后 ， 将 输出 如 下 所 示 的 信息 : 


3. is_writable() 
它 的 使 用 格式 如 下 所 示 : 


使 用 is_writable0 函 数 的 具体 示例 如 下 所 示 : 


上 述 代码 成 功 执行 后 ， 将 输出 如 下 所 示 的 信息 : 
ityingtxtiswitable 


4. umask() 
它 的 使 用 格式 如 下 所 示 : 


这 里 要 注意 ， 在 多 线程 的 服务 器 上 尽量 避免 使 用 这 个 函数 。 创 建文 件 后 要 改变 其 权限 最 好 使 用 
chmodO 函 数 。 使 用 umaskO 函 数 会 导致 并 发 程序 和 服务 器 发 生 不 可 预知 的 情况 ， 因 为 它们 是 使 用 相 
同 的 umask 的 。 


在 进行 PHP 程序 开发 时 ， 经 常 要 与 文件 和 数据 库 打 交道 。 本 节 主 要 介绍 关于 文件 方面 的 操作 ， 
主要 包括 文件 的 打开 和 关闭 、 读 取 文件 、 移 动 文件 指针 、 写 入 文件 、 读 取 目 录 内 容 及 其 他 相关 的 概 


9.3.1 文件 MO 基本 概念 


文件 IO 基本 概念 的 相关 知识 比较 多 ， 这 里 主要 介绍 资源 、 文 件 中 的 换行 及 文件 末尾 这 三 方面 
的 概念 ， 如 表 9-7 所 示 。 
表 9-7 文件 /O 基 本 概念 


文件 IO 基本 概念 说 明 

资源 资源 〈Iesource) 这 个 词 通常 与 可 以 发 起 输入 或 输出 流 的 实体 联系 在 一 起 。 标 准 的 输入 或 输 
出 、 文 件 和 网 络 套 接 字 都 是 资源 

换行 换行 符 通 过 字符 序列 \n 表示 ， 表 示 文 件 中 一 行 的 末尾 。 当 需要 一 次 输入 或 输出 一 行 信息 时 ， 
可 以 使 用 这 个 字符 序列 ， 如 file0、fgetcsv0 和 fgets0 等 函数 提供 了 处 理 换行 符 的 功能 

文件 末尾 程序 需要 一 种 标准 的 方式 来 识别 何 时 到 达 文 件 的 末尾 。 这 个 标准 通常 称 为 文件 末尾 或 EOF 


字符 。EOF 对 于 程序 设计 来 说 是 非常 重要 的 ， 因 为 几乎 所 有 的 主流 编程 语言 都 提供 了 相应 的 
内 置 函数 ， 来 验证 解析 器 是 否 到 达 了 文件 末尾 


在 PHP 中 ， 实 现 此 功能 的 函数 是 fgof0， 它 的 使 用 格式 如 下 所 示 : 


bool feof ( resource handle) 


该 函数 用 于 测试 文件 指针 是 否 到 了 文件 末尾 。 如 果 文 件 指针 到 了 EOF 或 者 出 错 则 返回 True, 否 
则 返回 False。 这 里 需要 注意 ， 文 件 指针 必须 有 效 ， 并 且 是 通过 fopen0、popen0 或 fsockopen0 函 数 
成 功 打 开 的 。feof0 函 数 的 具体 示例 如 下 所 示 : 


<?php 
$filename = "jack/itying.txt'; 
S$fe = fopen($filename, 'rt'); 
while (!feof ($fe)) 
{ 
echo fgets ($fe); 
1 
fclose(Sfe) 7 
?> 


上 述 代 码 成 功 执行 后 ， 将 输出 itying.txt 文件 的 内 容 。 


灌 弃 满 夫 着 半 峭 这 ”机 加 浊 


统 


191 


9.3.2 ”打开 和 关闭 文件 


建立 访问 文件 的 过 程 被 称 为 打开 文件 。 在 进行 读 或 写 一 个 文件 之 前 ， 必 须 先 打开 这 个 文件 。 另 
外 由 于 打开 的 文件 会 占用 系统 资源 ， 当 脚本 使 用 完 文件 后 ， 应 关闭 该 文件 ， 释 放 其 所 占用 的 资源 。 
PHP 会 在 脚本 执行 的 最 后 自动 关闭 打开 的 文件 ， 当 然 用 户 可 以 通过 使 用 相应 的 函数 来 提前 关闭 打开 


的 文件 。 
1. 打开 文件 
要 打开 文件 可 以 使 用 fopen0 函 数 ， 它 的 使 用 格式 如 下 所 示 : 
resource fopen ( string filename, string mode [, int use include path [, resource 


zcontext]]); 


该 函数 用 于 将 f 包 ename 指定 的 名 字 资 源 绑 定 到 一 个 流 上 。 如 果 flename 是 “scheme://...” 的 格 
式 ， 则 被 当 作 一 个 URL，PHP 将 搜索 协议 处 理 器 (也 被 称 为 封装 协议 ) 来 处 理 此 模式 。 如 果 该 协议 
尚未 注册 封装 协议 ，PHP 将 发 出 一 条 消息 来 帮助 检查 脚本 中 潜在 的 问题 ， 并 将 filename 当 作 一 个 普 
通 的 文件 名 继续 执行 下 去 。 

如 果 PHP 认为 filename 指定 的 是 一 个 本 地 文件 , 将 尝试 在 该 文件 上 打开 一 个 流 。 该 文件 必须 是 
PHP 可 以 访问 的 ， 因 此 需要 确认 文件 访问 权限 允许 该 访问 。 如 果 激 活 了 安全 模式 或 者 open_basedir， 
则 会 应 用 进一步 的 限制 。 

如 果 PHP 认为 filename 指定 的 是 一 个 已 注册 的 协议 (如 HTTP、HTTPS 和 FTP 等 )， 而 该 协议 
被 注册 为 一 个 网 络 URL，PHP 将 检查 并 确认 allow_url fopen 已 被 激活 。 如 果 关闭 了 ，PHP 将 发 出 
一 个 警告 ， 而 fopen0 函 数 的 调用 则 失败 。 

打开 resource 时 ,如 果 指 定 了 mode, 用 户 就 可 以 确定 该 资源 的 访问 级 别 。 如 表 9-8 所 示 为 mode 
可 以 指定 的 各 种 模式 。 
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表 9-8 mode 可 以 指定 的 模式 

模 式 说 有 明 

E 只 读 方式 打开 ， 将 文件 指针 指向 文件 头 

+ 读 写 方式 打开 ， 将 文件 指针 指向 文件 头 

Ww 写 入 方式 打开 ， 将 文件 指针 指向 文件 头 并 将 文件 大 小 截 为 零 。 如 果 文 件 不 存在 则 尝试 创建 

w+ ”| 读 写 方式 打开 ， 将 文件 指针 指向 文件 头 并 将 文件 大 小 截 为 零 。 如 果 文 件 不 存在 则 尝试 创建 

a 写 入 方式 打开 ， 将 文件 指针 指向 文件 末尾 。 如 果 文 件 不 存在 则 尝试 创建 

at 读 写 方式 打开 ， 将 文件 指针 指向 文件 末尾 。 如 果 文 件 不 存在 则 尝试 创建 

汉 创建 并 以 写 入 方式 打开 , 将 文件 指针 指向 文件 头 。 如 果 文件 已 存在 , 则 fopenO 函 数 调用 失败 并 返回 False， 
并 生成 一 条 E_WARNING 级 别 的 报错 信息 。 如 果 文件 不 存在 则 尝试 创建 。 此 选项 仅 能 用 于 本 地 文件 
x+ 创建 并 以 读 写 方式 打开 , 将 文件 指针 指向 文件 头 。 如 果 文件 已 存在 , 则 fopenO 函 数 调用 失败 并 返回 False， 
并 生成 一 条 E_WARNING 级 别 的 报错 信息 。 如 果 文 件 不 存在 则 尝试 创建 。 此 选项 仅 能 用 于 本 地 文件 


如 果 资 源 位 于 本 地 文件 ，PHP 则 认为 可 以 使 用 本 地 路 径 或 相对 路 径 来 访问 此 资源 。 或者， 可 以 
将 fopen0 函 数 的 use_include_path 参数 设置 为 1， 这 样 就 会 使 PHP 考虑 配置 指令 include_path 中 指 
定 的 路 径 。 

参数 zcontext 是 用 来 设置 文件 或 流 特 有 的 配置 参数 ， 以 及 在 多 个 fopenO 函 数 请 求 之 间 共 享 文件 
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或 流 特有 的 信息 。fopen0 函 数 的 具体 示例 如 下 所 示 : 


<?php 

$fpl = fopen ("jack/itying.txt", "r"); . 

$fp2 = fopen ("jack/itying.txt", "wb"); 3 

$fp3 = fopen ("http://www.itzcn.com/", "r"); 本 局 

$pf4 = fopen ("ftp://user:password@itying.net/text.txt", "w"); 2 

> 

在 Windows 平台 上 ， 要 小 心 转 义 文件 路 径 中 的 每 个 反 斜 线 或 者 斜 线 ， 如 下 所 示 : 

<?php 

$fp = fopen ("c:\\myfile\\info.txt", "r"); 

?> 第 
显 

2. 关闭 文件 

要 关闭 已 打开 的 文件 可 以 使 用 felose0 函 数 ， 它 的 使 用 格式 如 下 所 示 : 处 

bool fclose ( resource handle); 于 
件 

该 函数 用 于 将 handle 指向 的 文件 关闭 。 如 果 成 功 则 返回 True， 否 则 返回 False。 文 件 指针 必须 和 

有 效 ， 并 且 是 通过 fopen0 或 fockopen0 函 数 成 功 打 开 的 。fcloseO 函 数 的 具体 示例 如 下 所 示 : 寺 
<?php 风 
Shandle = fopen('somefile.txt', ‘'r'); 
fclose($handle); 
> 
9.3.3 读 取 文 件 


通常 情况 下 ， 在 打开 一 个 文件 以 后 就 能 读 取 该 文件 中 的 数据 了 。PHP 提供 了 许多 函数 来 完成 这 
一 功能 ， 不 仅 可 以 一 次 只 读 取 一 个 字符 ， 还 可 以 一 次 读 取 整个 文件 。 本 节 将 介绍 这 些 函 数 ， 其 详细 
信息 如 表 9-9 所 示 。 


表 9-9 读 取 文 件 函数 
函数 说 了 明 
file0 该 函数 用 于 把 整个 文件 读 入 一 个 数组 中 ， 各 元 素 由 换行 符 分 隔 ， 同 时 换行 符 仍 附加 在 每 个 元 


素 的 末尾 。 它 和 fle _get_contentsO 函 数 不 同 的 是 ， 它 将 文件 作为 一 个 数组 返回 。 如 果 失 败 则 
返回 False。 如 果 想 在 include_path 中 搜寻 文件 人 lename， 可 以 将 可 选 参数 use_include_ path 设 
i A 

file_get_contents() | 该 函数 用 于 将 整个 文件 读 入 一 个 字符 串 。 它 是 用 来 将 文件 的 内 容 读 入 一 个 字符 串 的 首选 方法 。 
如 果 操作 系统 支持 还 可 以 使 用 内 存 映 射 技术 来 增强 性 能 。 如 果 提 供 可 选 参 数 offset 和 maxlen， 
那么 它 将 在 参数 offset 所 指定 的 位 置 开 始 读 取 长 度 为 maxlen 的 内 容 。 如 果 失 败 则 返回 False 

TIeadfile0) 该 函数 用 于 读 取 由 fename 指定 的 整个 文件 , 立即 输出 到 输出 缓冲 区 , 并 返回 读 取 的 字 节 数 。 
以 @readfile0 形 式 调用 , 如 果 出 错 则 返回 False, 否则 将 显示 报错 信息 。 如果 启用 可 选 参数 use_ 
include_path， 将 告诉 PHP 在 配置 指令 include_path 指定 的 路 径 中 搜索 文件 

fscanfO) 该 函数 提供 了 一 种 可 以 按照 format 指定 的 格式 解析 由 handle 指定 的 资源 。 如 果 只 给 此 函数 传 
递 了 两 个 参数 ， 解 析 后 的 值 会 被 作为 数组 返回 。 否 则 ， 如 果 提 供 了 可 选 参 数 ， 此 函数 将 返回 
被 赋值 的 数目 。 可 选 参数 必须 用 引用 传递 
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续 表 
函数 说 了 明 


fgetcO 该 函数 用 于 从 文件 指针 中 读 取 字符 。 它 返回 一 个 包含 有 一 个 字符 的 字符 串 ， 该 字符 从 handle 指 
向 的 文件 中 得 到 ， 如 果 遇 到 EOF 则 返回 False。 这 里 需要 注意 ， 文 件 指针 必须 有 效 ， 并 且 必 须 
指向 一 个 由 fopen0 或 fockopenO 函 数 成 功 打开 《但 还 没有 被 人 lose0 函 数 关 闭 ) 的 文件 

feetcsvO 该 函数 用 于 解析 由 handle 指定 的 文件 中 的 每 一 行 , 各行 以 delimiter 分 隔 ， 并 把 各 个 部 分 放 在 数 
组 中 。 遇 到 换行 时 读 取 不 会 停止 ， 而 会 在 读 取 了 length 个 字符 后 ， 或 发 现 结束 字符 enclosure 
时 停止 。 因此， 应 该 选择 一 个 较 大 的 数值 ， 保 证 它 肯定 超过 文件 中 最 长 一 行 的 长 度 。 该 函数 出 
错 或 碰 到 文件 结束 时 返回 False 


feets0 该 函数 从 handle 指向 的 文件 中 读 取 一 行 并 返回 长 度 最 多 为 length-1 字 节 的 字符 串 。 遇 到 换行 符 
(包括 在 返回 值 中 )、EOF 或 者 已 经 读 取 了 length-1 字 节 后 停止 。 如 果 没 有 指定 length， 则 默认 


和 为 IKB (1024B)。 如 果 出 错 则 返回 False 
与 feetss0 该 函数 从 文件 指针 中 读 取 一 行 并 过 滤 掉 HTML 标记 。 它 和 他 ets0 函 数 唯 一 不 同 的 是 ， 人 eetssO 
三 函数 尝试 从 读 取 的 文本 中 去 掉 任何 HTML 和 PHP 标记 ， 但 可 以 用 可 选 的 第 三 个 参数 指定 哪些 
标记 不 被 去 掉 
2 freadO 该 函数 从 handle 指定 的 资源 中 读 取 length 个 字符 。 当 到 达 EOF 或 读 取 到 length 个 字符 时 ， 读 
完 取 将 停止 。 这 里 需要 注意 ， 该 函数 与 其 他 读 取 函 数 不 同 ， 使 用 它 不 需要 考虑 换行 符 ， 因 此 ， 只 
全 要 使 用 flesize0 函 数 确定 了 应 当 读 取 的 字符 数 ， 就 能 很 方便 地 使 用 这 个 函数 来 读 取 整 个 文件 
学 
下 下 面 分 别 对 表 9-9 所 示 的 函数 的 格式 进行 介绍 及 举例 。 
册 1. file() 

它 的 使 用 格式 如 下 所 示 : 


array file ( string filename [, int use include path [, resource context]]) 
假如 存在 一 个 文本 文件 weburltxt， 它 的 内 容 如 下 所 示 : 


WWW.itzcn.cCom 
Www.itzcn.net 
Www.baidu.com 
WWW .google.com 
www.itying.net 


下 面 的 代码 用 于 读 取 weburl.txt 文件 的 内 容 : 


<?php 
// 将 文件 weburl .txt 的 内 容 读 入 数组 
$lines = file('jack/weburl.txt"); 
// 在 数组 中 循环 显示 weburl .txt 文件 内 容 
foreach ($lines as $line num => S$line) 1{ 
echo "Line #<b>{$line num}</b> : <a href=" . $line.">". $line."</a> "7 
人 


上 述 代码 成 功 执 行 后 ， 输 出 内 容 的 源 代码 如 下 所 示 : 


Line #<b>0</b> : <a href=www.itzcn.com>www.itzcn.com</a> 
Line #<b>1</b> : <a href=www.itzcn.net>www.itzcn.net</a> 
Line #<b>2</b> : <a href=www.baidu.com>www.baidu.com</a> 
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2. file_get_contents() 
它 的 使 用 格式 如 下 所 示 : 


使 用 fle_get_contentsO) 函 数 的 具体 示例 如 下 所 示 : 


上 述 代码 成 功 执行 后 ， 输 出 的 内 容 与 weburl.txt 的 实际 内 容 一 样 ， 这 里 不 再 列 出 。 
3. readfile() 
它 的 使 用 格式 如 下 所 示 : 


使 用 readfile0 函 数 的 具体 示例 如 下 所 示 : 


上 述 代码 成 功 执行 后 ， 输 出 的 内 容 为 :75。 
4. fscanf() 
它 的 使 用 格式 如 下 所 示 : 


这 里 需要 注意 ， 格 式 字符 串 中 的 任何 空白 会 与 输入 流 中 的 任何 空白 匹配 ， 这 意味 着 甚至 格式 字 
符 串 中 的 制 表 符 “\t” 也 会 与 输入 流 中 的 一 个 空格 字符 匹配 。fscanf0 函 数 的 具体 示例 如 下 所 示 : 


5. fgetc() 
它 的 使 用 格式 如 下 所 示 : 


使 用 fgetc0 函 数 的 具体 示例 如 下 所 示 : 
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2 

全 上 述 代 码 成 功 执行 后 将 逐个 字符 输出 weburl.txt 的 内 容 。 
习 6. fgetcsv() 

外 它 的 使 用 格式 如 下 所 示 : 


使 用 fgetcsv0 函 数 的 具体 示例 如 下 所 示 : 


上 述 代 码 成 功 执行 后 ， 输 出 如 下 所 示 的 信息 : 


7. fgets() 
它 的 使 用 格式 如 下 所 示 : 


string fgets (int handle [, int length]); 


使 用 fgets0 函 数 的 具体 示例 如 下 所 示 : 


. 
. 
<?php 
$fp = @fopen("jack/weburl.txt", "r"); yg 
if ($fp) { 


while (!feof($fp)) { 
$fgs = fgets ($fp, 2048); 


echo $fgs; 
} 

fclose ($fp); 
. 第 
> 9 
草 
8. fgetss() 处 
它 的 使 用 格式 如 下 所 示 : 站 
string fgetss(resource handle [, int length [, string allowable tags]]); 全 
| 
9. fread() 7 
它 的 使 用 格式 如 下 所 示 : 系 


统 


string fread (int handle, int length ); 
使 用 freadO 函 数 的 具体 示例 如 下 所 示 : 


<?php 

$file = "jack/weburl.txt"; 

$fp = fopen($file, "r"); 

$contents = fread($fp, filesize ($file)); 
echo $contents; 

fclose ($fp); 

?> 


9.3.4 ”移动 文件 指针 


在 对 文件 进行 操作 时 ， 经 常 要 移动 文件 指针 的 位 置 来 对 文件 的 内 容 进行 读 写 。 为 了 实现 上 述 功 

能 ，PHP 提供 了 如 表 9-10 所 示 的 函数 。 
表 9-10 ”移动 文件 指针 函数 
函数 说 有明 
fseek( ”| 该 函数 用 于 设 定 handle 指定 文件 中 的 文件 指针 位 置 。 如 果 忽 略 可 选 参 数 whence， 则 位 置 将 设置 为 从 文 
件 开头 的 offset 字 节 处 计算 ， 否则 位 置 从 whence 指定 的 位 置 加 上 offset。 其中，whence 值 可 以 设置 为 : 
@ SEEK SET 设 定 指针 位 置 为 offset 字 节 处 。 如 果 没 有 指定 whence， 默 认为 该 值 。 
@ SEEK CUR 设 定 指针 位 置 为 当前 位 置 加 上 offset. 
e@ SEEK END 设 定 指针 位 置 为 文件 尾 (EOF) 加 上 offset。( 这 里 要 移动 到 文件 尾 之 前 的 位 置 ， 需 要 给 
offset 传递 一 个 负 值 ) 
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说 了 明 
该 函数 用 于 返回 由 handle 指定 的 文件 指针 的 位 置 , 也 就 是 文件 流 中 的 偏 移 量 。 如 果 出 错 , 则 返回 False。 
这 里 文件 指针 必须 是 有 效 的 ， 且 必须 指向 一 个 通过 fopen0 或 popen0 函 数 成 功 打开 的 文件 

IewindO | 该 函数 用 于 将 handle 的 文件 位 置 指针 设 为 文件 流 的 开头 。 如 果 成 功 则 返回 True， 和 否则 返回 False。 文件 
指针 必须 合法 ， 并 且 指向 由 fopenO 函 数 成 功 打开 的 文件 


下 面 分 别 对 表 9-10 所 示 的 函数 的 格式 进行 介绍 及 举例 。 
1. fseek() 
它 的 使 用 格式 如 下 所 示 : 


int fseek(resource handle, int offset [,int whence]); 
如 果 fseek0 函 数 执行 成 功 则 返回 0， 否则 返回 -1。 它 的 具体 示例 如 下 所 示 : 


<?php 

$fp = fopen('jack/weburl.txt', ‘'r'); 
$data = fgets($fp, 4096); 

// 移 动 文件 指针 

fseek ($fp, 0); 
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2. ftell() 
它 的 使 用 格式 如 下 所 示 : 


int ftell (resource $handle); 


使 用 ftell0 函 数 的 具体 示例 如 下 所 示 : 


<?php 

$fp = fopen("jack/weburl .txt", "r"); 
$data = fgets ($fp, 20); 

// 获取 并 输出 当前 指针 位 置 

echo ftell ($fp); 

fclose ($fp); 

?> 


3. rewind() 
它 的 使 用 格式 如 下 所 示 : 


bool rewind(resource $handle); 


9.3.5” 写 入 文件 


在 进行 程序 开发 时 ， 经 常 要 向 文件 中 写 入 数据 。 为 此 ，PHP 提供 了 一 些 函 数 来 实现 这 些 功 能 。 
下 面 将 分 别 进行 介绍 。 

1. fwrite() 

它 的 使 用 格式 如 下 所 示 : 
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该 函数 用 于 将 string 的 内 容 写 入 由 handle 指定 的 资源 中 。 如 果 给 出 可 选 参数 length， 该 函数 将 
在 写 入 了 length 个 字符 时 停止 ， 否 则 将 一 直 写 到 string 结尾 时 才 停止 。 该 函数 执行 结束 ， 返 回 写 入 
的 字符 数 ， 出 现 错误 时 则 返回 False。 

下 面 的 案例 用 于 向 write.txt 文件 写 入 一 些 信息 ， 有 具体 代码 如 下 所 示 : 


上 述 代 码 成 功 执行 后 ， 将 输出 如 下 所 示 的 信息 : 


2. fputs() 
它 的 使 用 格式 如 下 所 示 : 


int fputs(resource handle, string string [, int length); 
该 函数 与 fwrite0 函 数 的 功能 相同 。PHP 之 所 以 提供 这 个 函数 ， 可 能 是 为 了 与 C 及 C++ 保 持 


一 致 
9.3.6 读 取 目 录 内 容 


前 面 已 经 介绍 了 与 读 取 文件 相关 的 函数 , 这 里 将 介绍 一 些 读 取 目录 内 容 的 函数 , 如 表 9-11 所 示 。 
这 几 种 读 取 操 作 的 过 程 很 相似 。 
表 9-11 读 取 目 录 内 容 函 数 


函 数 使 用 格式 说 明 


该 函数 用 于 打开 一 个 目录 句柄 (好 比 文件 中 的 指针 )， 可 用 于 之 后 的 
closedir0 、readdir0 和 rewinddir0 函 数 调用 中 。 如 果 成 功 ， 则 返回 目录 句 
柄 的 resource; 否则 返回 False。 这 里 需要 注意 ， 如 果 path 不 是 一 个 合 
法 的 目录 或 者 因为 权限 限制 或 文件 系统 错误 而 不 能 打开 目录 ， 则 返回 
False 并 产生 一 个 E_WARNING 级 别 的 PHP 报错 信息 。 可 以 在 opendir0 
函数 前 面 加 上 “@” 符 号 来 抑制 报错 信息 的 输出 

该 函数 用 于 返回 目录 中 下 一 个 文件 的 文件 名 , 并 且 文 件 名 以 在 文件 系统 


opendir() | resource opendir(string path 
[resource $context]): 


readdir() | string readdir(resource 


dir handle); 中 的 排序 返回 。 如 果 成 功 则 返回 文件 名 ， 否 则 返回 False 
closedir() | void closedir(resource 该 函数 用 于 关闭 由 dir_ handle 指定 的 目录 流 。 该 目录 流 必 须 之 前 被 
S$dir handle): opendir0 函 数 打开 


该 函数 用 于 返回 一 个 数组 , 它 包 含 directory 中 的 文件 和 目录 。 如 果 成 功 
则 返回 包含 文件 名 的 数组 ， 否 则 返回 False。 如 果 directory 不 是 目录 ， 
则 返回 布尔 值 False 并 生成 一 条 E_WARNING 级 的 错误 。 默 认 的 排序 顺 
序 是 按 字母 升序 排列 。 如 果 使 用 了 可 选 参数 sorting order ( 设 为 1), 则 
排序 顺序 是 按 字 母 降序 排列 


以 下 案例 用 于 读 取 当 前 目录 及 内 容 ， 具 体 代码 如 下 所 示 : 
案例 9-4 


<pre> 

<?php 

Sir ms mep 

// 打 开 目 录 并 读 取 目 录 内 容 
if (is dir($dir)) 

{ 


scandir() | array scandir(string directory 
[Lint sorting_order [,resource 
context]]): 
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if ($dh = opendir($dir)) 
* 
while (($file = readdir($dh)) !== false) 
{ 
echo "filename: $file : filetype: " . filetype($dir . $file) . "\n"7 
} 
closedir ($dh); 


</pre> 


上 述 代码 成 功 执行 后 ， 输 出 类 似 于 如 下 所 示 的 信息 : 


使 用 scandir0 函 数 的 具体 示例 如 下 所 示 : 


上 述 代码 成 功 执行 后 ， 将 输出 类 似 于 如 下 所 示 的 信息 : 


9.4 执行 Shell 命令 


很 多 语言 都 提供 了 与 底层 操作 系统 进行 交互 的 能 力 , PHP 也 不 例外 , 本 节 将 介绍 这 方面 的 知识 。 
在 PHP 中 ,除了 可 以 使 用 exec0 或 system0 等 函数 来 执行 任何 系统 级 的 命令 外 , 还 可 以 使 用 如 表 9-12 
所 示 的 函数 。 
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表 9-12 Shell 命令 函数 


函数 使 用 格式 说 阴 
rmdir0 int rmdir(string 该 函数 用 于 删除 由 dimame 指定 的 目录 ， 如 果 成 功 则 返回 Trme， 否 则 返 
dimame): 回 False。 这 里 应 该 注意 ， 要 想 成 功 删除 目录 ， 必 须 正 确 地 设置 权限 。 因 


为 ， 通 常情 况 下 要 以 服务 器 的 守护 进程 所 有 者 来 执行 PHP 脚本 ， 所 以 如 
果 用 户 对 目录 没有 写 权限 ，mmdir0 函 数 将 执行 失败 ; 另外 ， 所 删除 的 目 
录 必 须 为 空 


rename() boolean rename(string | 该 函数 将 oldname 指定 的 文件 重 命名 为 新 名 newname， 如 果 成 功 则 返回 
oldname, string True， 和 否则 返回 False。 同 mmdir 一 样 ， 以 服务 器 守护 进程 所 有 者 来 执行 
newname): PHP 脚本 ， 所 以 在 用 户 没有 文件 的 写 入 权限 时 ，renameO 函 数 将 执行 失败 

touchO int touch (string 该 函数 设置 文件 filename 的 最 后 修改 时 间 和 最 后 访问 时 间 ， 如 果 成 功 则 
filename, [int 返回 Tme， 否 则 返回 False。 如 果 没 有 给 出 time， 则 使 用 当前 时 间 (由 服 
time[int atime]]): 务 器 指定 )。 如 果 给 出 了 可 选 参数 atime， 则 将 访问 时 间 设 置 为 这 个 值 ; 


否则 ， 与 修改 时 间 一 样 ， 将 设置 为 time 或 服务 器 时 间 。 这 里 应 该 注意 ， 
如 果 filename 不 存在 ， 而 且 脚本 所 有 者 具有 足够 的 权限 ， 那 么 将 创建 这 
个 文件 


用 户 可 以 通过 执行 系统 级 命令 ， 如 exec0 或 system0 〇 函数, 或 者 编写 一 个 递归 函数 ,在 删除 目录 
前 先 删 除 其 中 的 所 有 文件 。 这 里 需要 注意 ， 在 这 两 种 情况 下 ， 执 行 脚本 的 用 户 ( 服 务 器 守护 进程 所 
有 者 ) 都 需要 对 目标 目录 的 父 目录 具有 写 入 权限 。 以 下 语句 就 是 一 个 删除 目录 的 递归 函数 ， 它 的 具 
体 代 码 如 下 所 示 : 


<?php 
function delDir($dir) 
i 
if($0d = @opendir ($dir)) 
4 
while (($file = readdir(&0d)) != false) 
{ 
if($file == ".") || ($file =="..")) 
continue; 
if(is dir($dir.'/'.$file)) 
delDir($dir.'/'.$file); 
else 
unlink($0d.'/' .$file); 


L 


@closedir ($0d); 
rmadir (Sod) 7 


$dir = "test/"; 
delDir ($dir); 


9.5 ”系统 级 程序 执行 


在 PHP 中 ， 系 统 级 程序 执行 主要 是 利用 整个 服务 器 环境 ， 包 括 操作 系统 、 文 件 系统 、 已 安装 的 
程序 库 和 第 三 方 应 用 程序 来 加 快 PHP 编程 。 但 有 时 使 用 不 当 也 有 可 能 带 来 严重 的 后 果 , 所 以 在 使 用 
之 前 要 清理 输入 。 


9.5.1 清理 输入 


为 了 防止 系统 级 程序 执行 带 来 问题 ， 必 须 在 将 用 户 的 输入 传递 给 任何 PHP 程序 执行 之 前 , 对 输 
入 进行 清理 。 这 里 可 以 使 用 escapeshellarg0 和 escapeshellcemd0 两 个 标准 函数 来 实现 该 功能 。 

1. escapeshellarg() 

它 的 使 用 格式 如 下 所 示 : 

string escapeshellarg (string argument); 

该 函数 用 单 引号 界定 argument， 并 将 argument 中 的 单 引 号 加 上 前 缀 〈 转 义 )， 也 就 是 说 ， 当 把 
argument 传递 给 Shell 命令 时 ， 会 把 它 认 为 是 单个 参数 ， 因 为 这 样 减少 了 攻击 者 利用 Shell 命令 参数 
伪装 额外 命令 的 可 能 性 。 因 此 ， 对 于 上 述 情况 ， 整 个 用 户 输入 会 包围 在 单 引号 中 ， 具 体 如 下 所 示 : 


"http://www.itzcn.com/ ; cd /usr/local/apache/htdoc/; rm -rf *" 

执行 上 述 语句 ，HIMLDOC 将 返回 一 个 错误 ， 因 为 它 无 法 解析 有 这 种 语法 的 URL， 而 不 是 删除 
整个 目录 。 

2. escapeshellcmd() 

它 的 使 用 格式 如 下 所 示 : 

string escapeshellcmd (string command); 


该 函数 的 工作 过 程 与 escapeshellarg0 相 同 ， 通 过 对 Shell 元 字符 转 义 来 清理 可 能 危险 的 输入 。 这 
些 字 符 包 搬 失语 下 于 2B 


9.5.2 PHP 的 程序 执行 函数 


关于 exec0 及 system0 函 数 前 面 已 经 多 次 提 及 ， 本 章 就 介绍 这 些 通 过 PHP 脚本 执行 系统 程序 的 
函数 ， 如 表 9-13 所 示 。 


表 9-13 ”PHP 的 程序 执行 函数 


该 函数 用 于 在 服务 器 后 台 连 接 执行 操作 系统 级 应 用 程序 (通过 command 指定 )。 虽 然 它 会 返回 和 输 
出 的 最 后 一 行 ， 但 用 户 通常 情况 下 希望 得 到 所 有 输出 ， 所 以 可 以 通过 包括 可 选 参数 output 来 实现 ， 
它 将 包含 由 execO 函 数 指定 的 命令 结束 时 的 每 一 行 输出 。 另 外 ， 可 以 通过 包括 可 选 参数 Tetum_ var 
来 得 到 执行 命令 的 返回 状态 


systemO 当 用 户 想 输出 执行 命令 的 结果 时 可 以 使 用 systemO 函 数 
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函 数 说 了 明 
passthruO) 该 函数 与 exec0 函 数 相似 ， 不 同 的 是 passthru0 函 数 用 于 向 调用 者 返回 二 进 制 
shell execQO | 该 函数 提供 了 与 反 引号 相同 的 语法 形式 ， 它 用 于 执行 一 个 Shell 命令 ， 并 返回 输出 


反 引 号 使 用 反 引 号 〈backtick) 界定 字符 串 时 ， 就 是 告诉 PHP 该 字符 串 应 当 作为 Shell 命令 来 执行 ， 并 返 
回 所 有 输出 。 这 里 需要 注意 的 是 ， 反 引号 不 是 单 引号 ， 而 是 一 个 倾斜 的 引号 ， 在 常用 的 键盘 上 一 
般 与 波浪 线 (~) 在 同一 个 按键 上 


下 面 分 别 对 表 9-13 所 示 的 函数 的 格式 进行 介绍 及 举例 。 
1. exec() 


它 的 使 用 格式 如 下 所 示 : 


string exec (string command[,array output[,int return var]]); 


下 面 的 代码 是 Perl 脚本 ， 它 用 于 显示 目录 列表 ， 相 当 于 Windows 中 执行 一 个 dir 命令 ， 具 体 代 
码 如 下 所 示 : 


#! /usr/bin/perl 
my @languages = qw[perl php Python java c]; 
foreach $language (@languages){ 
print $language."<br/>"; 
} 


把 上 述 代 码 存放 在 文件 language.pl 中 。 该 Perl 脚本 很 简单 ， 如 果 在 Linux 平台 下 ， 可 以 立即 
运行 此 示例 ， 因 为 每 个 Linux 版 本 中 都 安装 有 Perl。 如 果 在 Windows 平台 下 ， 就 要 检查 是 否 有 
ActiveState (http://www.activestate.com) 的 ActivePerl 发 行 包 。 

下 面 的 PHP 代码 只 是 用 于 调用 该 Perl 脚本 ， 并 指明 将 输出 放 在 数组 result 中 ， 然 后 将 result 的 
内 容 输出 到 浏览 器 。 具 体 代 码 如 下 所 示 : 

<?php 

Soutcome = exec("language.pl", $results); 
foreach (Sresults as $result) 


echo $result; 
>: 


成 功 执行 上 述 PHP 代码 ， 具 体 结果 如 下 所 示 : 


perl 
php 
python 
java 

c 


2. system() 
它 的 使 用 格式 如 下 所 示 : 


string system(string command [,int return var]); 


该 函数 不 像 exec0 函 数 那 样 通过 可 选 参 数 返 回 输出 ， 而 是 直接 将 输出 返回 给 调用 者 。 但 是 ， 如 


果 用 户 想 查看 被 调用 程序 的 执行 状态 ， 就 需要 使 用 可 选 参数 return_var 指定 一 个 变量 。 例 如 ， 如 果 
用 户 想 输 出 位 于 某 特定 目录 中 的 所 有 文件 ， 可 以 使 用 如 下 所 示 的 代码 ;: 


或 者 修改 前 面 的 PHP 脚本 ， 再 次 使 用 system0O 函 数 调用 的 language.pl 来 实现 该 功能 。 具 体 代 码 
如 下 所 示 : 


3. passthru() 
它 的 使 用 格式 如 下 所 示 : 


例如 ， 假 设 希望 在 浏览 器 显示 GIF 图 片 前 ， 先 将 GIF 图 片 转换 为 PNG 格式 图 片 。 为 了 实现 该 
功能 ， 可 以 使 用 Netpbm 图 像 包 ， 它 遵循 GPL 许可 ， 用 户 可 以 从 http://netpbm.sourcefore.net 获得 相 
关 的 信息 。 有 具体 代码 如 下 所 示 : 


4. shell_exec() 
它 的 使 用 格式 如 下 所 示 : 


使 用 shell_execO 函 数 的 具体 示例 如 下 所 示 : 


5. 反 引 号 
使 用 反 引 号 的 具体 示例 如 下 所 示 : 


如 果 成 功 执行 上 述 语句 ， 将 返回 类 似 于 如 下 所 示 的 结果 : 
”The server timestamp is Sun Jun 26 18:30:21 ET 2007 
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在 网 站 建设 中 ，PHP 起 着 核心 的 作用 ， 它 是 用 户 和 数据 交互 的 中 间 桥 梁 。 在 网 站 中 还 有 另外 一 
个 非常 重要 的 基础 ， 就 是 数据 库 。 网 站 的 后 端 数据 库 放 置 着 网 站 所 有 的 数据 ， 因 此 只 有 掌握 好 数据 
库 技术 ， 才 能 够 构建 强大 的 网 站 。MySQL 是 一 种 开放 源 代码 的 关系 型 数据 库 管理 系统 (RDBMS )， 
MySQL 数据 库 系统 使 用 最 常用 的 数据 库 管理 语言 一 一 结构 化 查询 语言 (SQL) 进行 数据 库 管理 。 

phpMyAdmin 是 一 个 用 PHP 编写 的 、 可 以 通过 互联 网 控制 和 操作 MySQL 的 Web 应 用 程序 。 通 
过 phpMyAdmin 可 以 完全 对 数据 库 进 行 操作 ， 例 如 建立 、 复 制 、 删 除数 据 等 。 有 了 phpMyAdmin 就 
可 以 完全 不 使 用 MySQL 命令 ， 直 接 使 用 phpMyAdmin 就 能 管理 MySQL 的 所 有 数据 和 数据 库 。 


“和 内 容 摘要 1mwewsct 


掌握 MySQL 数据 库 的 登录 、 权 限 管理 及 用 户 管理 
掌握 如 何 创建 数据 库 、 表 及 索引 

掌握 如 何 备份 和 恢复 数据 库 

掌握 如 何 使 用 MySQL 数据 库 

理解 事务 的 概念 及 其 创建 

理解 存储 过 程 的 概念 及 其 创建 

掌握 使 用 MySQL Administrator 管理 数据 库 

掌握 使 用 phpMyAdmin 管理 数据 库 


10.1 ”MySQL 应 用 基础 


MySQL 关系 型 数据 库 使 用 系统 核心 提供 的 多 线程 机 制 提供 完全 的 多 线程 运行 模式 ， 提 供 了 面 
向 C、C++、Eiffel、Java、Perl、PHP 以 及 Python 等 编程 语言 的 编程 接口 API)， 支 持 多 种 字段 类 
型 ， 并 且 提供 了 完整 的 操作 符 支持 查询 中 的 SELECT 和 WHERE 操作 。 

本 节 主 要 介绍 MySQL 数据 库 的 安装 、 配 置 及 登录 、 用 户 信息 的 管理 、 数 据 库 和 表 的 创建 及 查 
看 ，MySQL 数据 库 的 权限 管理 及 数据 库 的 备份 与 恢复 等 。 


10.1.1 安装 配置 MySQL 


本 节 重 点 介绍 Windows 平台 下 的 MySQL 二 进 制 安装 过 程 及 其 配置 。 建 议 读者 下 载 二 进 制 安装 
版 本 ， 这 样 可 以 在 安装 过 程 中 对 MySQL 数据 库 进 行 一 些 配置 。 这 里 下 载 的 是 MySQL 的 5.0 版 本 。 


安装 过 程 如 下 所 示 : 
(1) 把 下 载 的 安装 程序 解压 后 ， 双 击 setup.exe 文件 ， 会 显示 MySQL 的 欢迎 界面 ， 单 击 Next 
按钮 ， 会 显示 【选择 安装 类 型 】 界 面 ， 如 图 10-1 所 示 。 
(2) 这 里 选择 Custom 安装 类 型 ， 以 便 更 改 安装 目录 及 要 安装 的 MySQL 组 件 ， 然 后 单 击 Next 
按钮 ， 会 显示 【选择 安装 组 件 】 界 面 ， 如 图 10-2 所 示 。 省 
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图 10-1 【选择 安装 类 型 】 界 面 图 10-2 【选择 安装 组 件 】 界 面 


(3) 选择 安装 所 有 组 件 ， 并 更 改 安装 路 径 为 D 盘 ， 然 后 单 击 Next 按钮 ， 会 显示 【准备 安装 】 
界面 ， 如 图 10-3 所 示 。 

(4) 单 击 Install 按钮 开始 安装 。 安 装 完成 后 ， 会 显示 一 个 让 用 户 创建 MySQL.com 账号 的 界面 ， 
如 图 10-4 所 示 。 用 户 可 以 选中 Skip Sign_Up 单 选 按钮 跳 过 这 一 步 。 单 击 Next 按钮 会 显示 【完成 安 
装 】 界 面 ， 如 图 10-5 所 示 。 
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10-3 【准备 安装 】 界 面 10-4 【创建 MySQL.com 账号 】 界 面 


(5) 选中 Configure the MySQL Server now 复 选 框 ， 单 击 Finish 按钮 ， 会 显示 MySQL 配置 向 导 
的 【欢迎 】 界 面 ， 单 击 Next 按钮 ， 会 显示 【选择 配置 类 型 】 界 面 ， 如 图 10-6 所 示 。 

(6) 选中 Detailed Configuration 单 选 按钮 ， 单 击 Next 按钮 ， 会 显示 【选择 服务 器 类 型 】 界 面 ， 
如 图 10-7 所 示 。 
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图 10-5 【完成 安装 】 界 面 
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10-6 【选择 配置 类 型 】 界 面 


(7) 选中 Developer Machine 单 选 按钮 ， 单 击 Next 按钮 ， 会 显示 【选择 数据 库 使 用 】 界 面 ， 如 


图 10-8 所 示 。 
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图 10-7 【选择 服务 器 类 型 】 界 面 
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图 10-8 【选择 数据 库 使 用 】 界 面 


(8) 选中 Multifunctional Database 单 选 按钮 ， 单 击 Next 按钮 ， 会 显示 【选择 数据 库 文件 的 存储 


路 径 】 界 面 ， 如 图 10-9 所 示 。 


(9) 保持 默认 的 选项 ， 然 后 单 击 Next 按钮 ， 


图 10-10 所 示 。 
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图 10-9 【选择 数据 库 文件 的 存储 路 径 】 界 面 


会 显示 【选择 并 发 连接 到 服务 器 的 数目 】 窗 口 ， 如 
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10-10 【选择 并 发 连接 到 服务 器 的 数目 】 界 面 


(10) 保持 默认 的 选项 ， 然 后 单 击 Next 按钮 ， 会 显示 【设置 网 络 选项 】 界 面 ， 如 图 10-11 所 示 。 
(11) 设置 端口 号 为 3306 或 其 他 未 曾 使 用 的 端口 号 ， 以 及 决定 是 否 启用 Enable Strict Mode 复 选 
框 。 然 后 单 击 Next 按钮 ， 会 显示 【选择 默认 的 字符 集 】 界 面 ， 如 图 10-12 所 示 。 
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图 10-11 【设置 网 络 选项 】 界 面 图 10-12 【选择 默认 的 字符 集 】 界面 


(12) 在 默认 情况 下 ，MySQL 使 用 的 是 latinl 字符 集 ( 即 ISO8859-1)， 为 了 支持 中 文 ， 这 里 选 
择 gb2312， 然 后 单 击 Next 按钮 ， 会 显示 【选择 Windows 选项 】 界 面 ， 如 图 10-13 所 示 。 

(13) 设置 MySQL 服务 器 作为 Windows 的 服务 运行 ，Windows 启动 时 ， 将 自动 运行 MySQL。 
然后 选中 Include Bin Directory in Windows PATH 复 选 枉 ， 便 将 MySQL 安装 目录 下 的 bin 目录 添加 
到 PATH 环境 变量 中 ， 这 样 就 可 以 在 任意 目录 下 执行 MySQL 提供 的 工具 了 。 单 击 Next 按钮 ， 会 显 
示 【 设 置 安全 选项 】 界 面 ， 如 图 10-14 所 示 。 
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图 10-13 【选择 Windows 选项 】 界 面 10-14 【设置 安全 选项 】 界 面 


(14) 设置 MySQL 内 置 的 root 用 户 的 密码 为 “123456789”， 单 击 Next 按钮 ， 会 显示 【准备 执 
行 】 界 面 ， 单 击 Execute 按钮 开始 执行 ， 执 行 完 后 单 击 Finish 按钮 完成 MySQL 服务 器 的 配置 。 


10.1.2 ”登录 到 数据 库 


在 安装 完成 MySQL 以 后 就 可 以 登录 到 数据 库 了 , 单 击 【 开 始 】 菜 单 , 选择 【所 有 程序 ]IMySQLI 
MySQL Server $.0IMySQL Command Line Client 命令 , 会 显示 MySQL Command Line Client 窗口 , 在 
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Enter password: 后 输入 前 面 设置 的 密码 
“123456789“， 然 后 按 Enter 键 ， 如 果 密 码 正 
确 即 可 登录 到 数据 库 。 如 图 10-15 所 示 为 登 
录 成 功 后 , 查看 MySQL 数据 库 中 的 所 有 数据 
库 的 情况 。 

如 果 登 录 失 败 , 可 以 同时 按 Ctrl+AltrDel 
键 ， 会 显示 【Windows 任务 管理 器 】 窗 口 ， 
查看 进程 中 MySQL 的 服务 进程 是 否 启动 。 
或 者 ， 从 【开始 】 菜 单 上 选择 【控制 面板 】 
命令 ， 会 显示 【控制 面板 】 窗 口 ， 然 后 双击 国 W999 贡 呈 村 要 后 全 而 数 挫 谢 
【管理 工具 】 图 标 ， 会 显示 【管理 工具 】 窗 口 ， 再 双击 【服务 】 图 标 ， 会 显示 【服务 】 窗 口 ， 确 认 
MySQL 服务 是 否 启动 ， 如 图 10-16 所 示 。 


ET 
a 


者 称 | 换 状 | 启动 类 型 


Buc Disk Nanager 直 测 i 点 视 新 天 已 启动 自动 


Maaserr hsinistrative Service 二 动 


图 10-16 查看 MySQL 服务 


10.1.3 ”修改 用 户 密码 


在 安装 配置 MySQL 数据 库 时 设置 了 root 账 号 (管理 员 ) 的 密码 为 “123456789“， 如 果 用 户 想 
修改 这 个 密码 可 以 使 用 SET PASSWORD 命令 ， 有 具体 如 下 所 示 : 


mysql>SET PASSWORD FOR roote@localhost=PRSSWORD("pwad123456")7 


通过 执行 上 面 的 设置 ， 当 用 户 退 出 当前 登录 ， 再 次 登录 时 ， 已 经 需要 使 用 密码 “pwd123456”， 
使 用 以 前 的 密码 将 造成 登录 失败 。 
出 于 安全 方面 的 考虑 ，MySQL 中 除了 root 管理 员 账 号 外 ， 还 有 许多 其 他 的 用 户 账号 ， 他 们 中 
每 一 个 用 户 赋 于 对 不 同 数据 库 的 访问 限制 ， 以 满足 不 同 用户 的 要 求 。 修 改 这 些 用 户 的 密码 主要 有 以 
下 几 种 方法 。 
1. 使 用 phpMyAdmin 
使 用 phppMyAdmin (图 形 化 管理 MySQL 数据 库 的 工具 )， 这 是 最 简单 的 ， 直 接 用 SQL 语句 修 
改 MySQL 数据 库 的 user 表 ， 这 里 要 使 用 PASSWORD 函数， 插入 用 户 使 用 Insert 命令 ， 修 改 用 户 
使 用 Update 命令 ， 删 除 用 户 使 用 Delete 命令 。 

2. 使 用 mysqladmin 

使 用 mysqladmin 输入 如 下 代码 : 


mysqladmin -u root -p oldpassword newpasswd 


执行 这 个 命令 后 , 需要 输入 root 的 原 密码 “oldpassword”, 这 样 root 的 密码 将 改 为 “newpasswd”。 
如 果 当 前 用 户 没 有 权限 执行 mysqladmin， 那 么 这 种 方法 就 是 无 效 的 ， 而 且 mysqladmin 无 法 把 密码 
清空 。 

3. 使 用 GRANT 语句 

使 用 GRANT...IDENTIFIED BY 语句 来 进行 授权 。 有 具体 如 下 所 示 : 


mysql> GRANT USAGE ON *.* TO root@loalhost IDENTIFIED BY "123456"7 


除了 前 面 的 方法 ， 另 外 还 可 以 使 用 前 面 修改 root 账号 的 方法 ， 也 就 是 使 用 SET PASSWORD 方 
法 进行 修改 。 

这 里 需要 说 明 的 是 ， 通 常情 况 下 ， 修 改 MySQL 密码 需要 有 MySQL 中 的 root 权限 ， 所 以 一 般 
用 户 是 无 法 更 改 密码 的 ， 除 非 请 求 管理 员 帮 助 修改 。 


10.1.4 ”MySQL 的 权限 管理 


MySQL 的 权限 管理 是 通过 GRANT 和 REVOKE 命令 来 实现 的 。 它 们 主要 用 于 对 操作 服务 器 及 
其 内 容 进 行 控制 ， 如 哪个 用 户 可 以 关闭 服务 器 。 哪 个 用 户 可 以 修改 数据 库 、 表 及 表 中 字段 等 。 如 表 
10-1 所 示 为 这 些 命令 可 以 授予 或 撤销 的 所 有 权限 。 
表 10-1 GRANT 和 REVOKE 管 理 的 权限 


权 限 说 阴 
ALL PRIVILEGES 影响 除 WITH GRANT OPTION 之 外 的 所 有 权限 
ALTER 影响 ALTER TABLE 命令 的 使 用 
CREATE 影响 CREATE TABLE 命令 的 使 用 
CREATE TEMPORARY TABLES 影响 CREATE TEMPORARY TABLE 命令 的 使 用 
CREATE VIEW 影响 CREATE VIEW 命令 的 使 用 
DELETE 影响 DELETE 命令 的 使 用 
DROP 影响 DROP TABLE 命令 的 使 用 
EXECUTE 影响 用 户 运 行 存储 过 程 的 能 力 
FILE 影响 SELECT INTO OUTFILE 和 LOAD DATA INFILE 的 使 用 
GRANT OPTION 影响 用 户 委派 权限 的 能 力 
INDEX 影响 CREATE INDEX 和 DROP INDEX 命令 的 使 用 
INSERT 影响 INSERT 命令 的 使 用 
LOCK TABLES 影响 LOCK TABLES 命令 的 使 用 
PROCESS 影响 SHOW PROCESSLIST 命令 使 用 
REFERENCES 未 来 MySQL 特性 的 占 位 符 
RELOAD 影响 FLUSH 命令 集 的 使 用 
REPLICATION CLIENT 影响 用 户 查询 从 服务 器 和 主 服 务 器 位 置 的 能 力 
REPLICATION SLAVE 复制 从 服务 器 所 需 的 权限 
SELECT 影响 SELECT 命令 的 使 用 
SHOW DATABASES 影响 SHOW DATABASES 命令 的 使 用 
SHOWVIEW 影响 SHOW CREATE VIEW 命令 的 使 用 
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权 限 说 了 明 
SHUTDOWN 影响 SHUTDOWN 命令 的 使 用 
SUPER 影响 管理 员 级 命令 的 使 用 , 如 CHANGE、 MASTER、 mysqladmin debug、 SET 
GLOBAL、KILL thread 及 PURGE MASTER LOGS 等 
UPDATE 影响 UPDATE 命令 的 使 用 
USAGE 只 连接 ， 不 授予 权限 


表 10-1 列 出 了 GRANT 和 REVOKE 的 管理 权限 ， 下 面 结合 示 例 来 介绍 这 两 个 命令 是 如 何 进行 
用 户 或 用 户 组 权限 设置 及 撤销 的 。 

1. GRANT 

GRANT 命令 用 于 为 用 户 或 用 户 组 赋予 新 的 权限 ， 它 的 具体 使 用 格式 如 下 所 示 : 


GRANT priv type [(column list)] [, priv type [(column list)] ...] 
ON {tbl name | * | *.* | db name.*} 
TO user name [IDENTIFIED BY [PASSWORD] ‘'password'] 
[, user name [IDENTIFIED BY "password'] ...] 
[REQUIRE 
NONE | 
[{SSLI X509}] 
[CIPHER cipher [AND]] 
[ISSUER issuer [AND]] 
[SUBJECT subject]] 
[WITH [GRANT OPTION | MAX QUERIES PER HOUR # | 
MAX UPDATES PER HOUR # | 
MAX_ CONNECTIONS PER HOUR #]] 


上 面 的 GRANT 命令 的 使 用 格式 看 起 来 很 复杂 ， 但 在 实际 应 用 中 ， 这 种 格式 很 少见 。 下 面 通过 
一 些 示 例 来 加 深 对 这 个 命令 的 掌握 。 

(1) 创建 新 用 户 

这 里 使 用 GRANT 命令 创建 一 个 新 用 户 ， 并 为 其 赋予 一 些 数据 库 特 定 的 权限 。 用 户 YanBob 从 
IP 地 址 192.168.1.11， 使 用 密码 “123456” 连 接 到 数据 库 服务 器 ， 并 拥有 对 student 数据 库 中 所 有 表 
的 SELECT 和 INSERT 权限 。 上 有 具体 语句 如 下 所 示 : 

mysql> GRANT select,insert,ON student.* TO YanBob@192.168.1.11 

-> IDENTIFIED BY "123456'7 


执行 上 述 语句 后 ， 将 修改 两 个 权限 表 ， 即 user 和 db 表 。 因 为 user 表 负 责 访问 验证 和 全 局 权限 ， 
所 以 必须 插入 一 个 新 记录 来 标识 此 用 户 。 由 于 GRANT 命令 只 作用 于 student 数据 库 ，db 表 将 包含 
相关 的 用 户 信息 ， 将 用 户 YanBob 映射 到 student 表 ， 并 启用 Select priv 和 Insert_priv 字段 。 

(2) 向 现 有 用 户 增加 权限 

假设 用 户 YanBob 需要 对 student 数据 库 中 的 所 有 表 进 行 UPDATE 操作 , 那么 必须 使 YanBob 拥 
有 这 个 权限 。 具 体 语句 如 下 所 示 : 


mysql> GRANT update ON student.* TO YanBob@192.168.1.11; 


执行 上 述 语句 后 ，db 表 中 标识 用 户 YanBob@192.168.1.11 的 记录 会 被 修改 ， 启 用 Update_priv 
字段 。 这 里 需要 注意 ， 向 现 有 用 户 增加 权限 时 不 需要 提供 密码 。 
(3) 授予 表 级 权限 


假设 用 户 YanBob@192.168.1.11 需要 对 student 数据 库 中 的 两 个 表 english 和 math 表 有 DELETE 
权限 ， 这 里 应 该 注意 ， 并 非 完全 允许 此 用 户 删除 student 数据 库 中 任何 表 的 数据 ， 而 是 通过 设置 权限 % 


来 限制 此 用 户 只 能 对 这 两 个 表 执 行 DELETE 操作 。 这 里 因 涉 及 两 个 表 , 所 以 需要 两 个 GRANT 命令 。 
具体 语句 如 下 所 示 : 


mysql> GRANT delete ON student.english TO YanBob@192.168.1.11; 


Query OK, 0 rows affected (1.52 sec) 
mysql> GRANT delete ON student.math TO YanBob@192.168.1.11; 
Query OK, 0 rows affected (1.12 sec) 


上 述 语 句 是 表 特定 的 权限 设置 ， 所 以 只 触及 tables_priv 表 。 执 行 上 述 语句 后 ， 如 果 之 前 没有 将 
english 表 和 math 表 映 射 到 YanBob@192.168.1.11 的 记录 ， 那 么 将 向 tables_priv 表 增 加 两 条 新 记录 。 
如 果 已 经 存在 这 样 的 记录 ， 则 会 相应 地 修改 现 有 的 记录 ， 反 映 这 两 个 表 特定 的 新 权限 。 

(4) 授予 多 个 表 级 权限 

授予 多 个 表 级 权限 是 为 用 户 提 供 针 对 一 个 给 定 表 的 多 个 权限 。 现 假设 一 个 新 用 户 SongYan 从 
itying.net 域 中 的 多 个 地 址 连接 ， 要 求 更 新 英语 信息 ， 那 么 只 需要 该 用 户 拥有 english 表 的 SELECT、 
INSERT 和 UPDATE 权限 就 行 了 。 授 予 用 户 SongYan 该 权限 的 具体 语句 如 下 所 示 : 


mysql> GRANT select, insert,update ON student.english TO SongYan@'%.itying.net' 
-> IDENTIFIED BY '123456'; 


执行 上 述 语句 后 ， 将 在 MySQL 数据 库 中 增加 两 个 新 项 : user 表 中 将 有 一 个 新 记录 用 于 为 
SongYan(@'%. itying.net 提供 访问 权限 ，tables_priv 表 中 也 将 有 一 条 新 记录 ， 指 定 用 于 student 表 的 新 
访问 权限 。 由 于 这 些 权限 只 应 用 于 一 个 表 , 所 以 只 向 tables_priv 表 增加 了 一 条 记录 , 其 中 , Table priv 
字段 值 为 Select.Insert.Delete。 

(5) 授予 字段 级 权限 

授予 字段 级 权限 只 影响 表 中 列 级 权限 。 现 假设 向 用 户 Jack@192.168.1.16 授予 student.english. 
score 的 UPDATE 权限 。 有 具体 语句 如 下 所 示 : 


mysql> GRANT update (Score) ON student.english TO Jack@192.168.1.16; 


2. REVOKE 
REVOKE 命令 负责 删除 之 前 授予 用 户 或 用 户 组 的 权限 。 它 的 具体 使 用 格式 如 下 所 示 : 
REVOKE priv type [(column list)] [, priv type [(column list)] ...] 

ON {tbl name | * | *-# | db name.*} 

FROM user name [, user name ...] 


通过 使 用 


下 面 通过 一 些 示 例 来 加 深 对 这 个 命令 的 掌握 。 
(1) 撤销 以 前 授予 的 权限 
既然 可 以 授予 用 户 某 些 特定 权限 ， 也 就 可 以 撤销 这 些 授予 的 权限 。 现 假设 要 撤销 用 户 
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YanBob@192.168.1.11 对 数据 库 student 的 UPDATE 权限 。 具 体 语句 如 下 所 示 : 


mysql> REVOKE update ON student.* FROM YanBob@192.168.1.11; 


(2) 撤销 表 级 权限 
现 假 设 要 撤销 之 前 向 用 户 YanBob@192.168.1.11 授予 的 对 数据 库 sudent 中 表 english 的 UPDATE 
和 JINSERT 权限 (用 户 YanBob@192.168.1.11 应 已 经 被 授予 了 表 级 权限 )。 具 体 语句 如 下 所 示 : 


mysql> REVOKE update,insert ON student.english FROM YanBob@192.168.1.11; 


这 里 要 注意 ，REVOKE 命令 不 会 将 数据 库 级 GRANT (位 于 db 表 ) 降级 ， 即 不 会 删除 db 表 中 
的 项 ， 而 在 tables_priv 表 中 插入 一 项 ， 所 以 在 这 种 情况 下 ， 它 只 是 删除 tables_priv 表 中 的 这 些 权限 
引用 。 如 果 tables_priv 表 中 只 引用 了 这 两 个 权限 ， 则 删除 整 条 记录 。 

(3) 撤销 列 级 权限 

现 假设 之 前 已 经 向 用 户 Jack@192.168.1.16 针 对 student.english 的 字段 score 授 予 了 列 级 UPDATE 
权限 ， 现 在 要 撤销 此 权限 。 具 体 语句 如 下 所 示 : 


mysql> REVOKE insert(score) ON student.english FROM Jack@192.168.1.16; 


(4) 撤销 用 户 所 有 权限 
有 时 需要 撤销 某 一 用 户 的 所 有 权限 ， 那 么 可 以 使 用 类 似 于 如 下 所 示 的 语句 : 


mysql> REVOKE all ON student.* FROM Jack@192.168.1.16; 


3. GRANT 和 REVOKE 的 相关 问题 

在 使 用 GRANT 和 REVOKE 命令 的 过 程 中 应 该 了 解 以 下 相关 问题 : 

(1) 用 户 可 以 为 不 存在 的 数据 库 授 权 。 

(2) 如 果 GRANT 标识 的 用 户 不 存在 ， 它 将 被 创建 。 

(3) 如 果 创 建 一 个 用 户 而 没有 包括 IDENTIFIED BY 子 句 ， 则 不 需要 密码 就 能 登录 。 

(4) 如 果 现 有 的 用 户 被 授予 新 权限 ， 并 且 在 GRANT 命令 中 使 用 IDENTIFIED BY 子 句 ， 则 该 
用 户 的 旧 密 码 将 被 新 密码 代替 。 

(5) 表 级 GRANT 只 支持 如 下 所 示 的 一 些 权限 类 型 : INSERT、CREATE、CREATE VIEW、 
DELETE、DROP、GRANT、INDEX、REFERENCES、SELECT、SHOW VIEW 和 UPDATE。 

(6) 列 级 GRANT 只 支持 如 下 所 示 的 一 些 权 限 类 型 : INSERT、SELECT 和 UPDATE。 

(7) 在 GRANT 命令 中 引用 数据 库 名 和 主机 名 时 ， 支 持 _ 和 % 通 配 符 。 因 为 字符 在 MySQL 数 
据 库 名 中 也 是 合法 字符 ， 所 以 如 果 在 GRANT 中 用 到 ， 需 要 用 反 斜 线 进行 转 义 。 

(8) 如 果 希 望 创建 和 删除 用 户 ， 而 且 运行 的 是 MySQL 5.0.2 或 更 高 版 本 ， 可 以 考虑 使 用 CREATE 
USER 和 DROP USER 命令 。 

(9) 不 能 引用 *.* 来 删除 某 用 户 对 所 有 数据 库 的 权限 。 必 须 用 单独 的 REVOKE 命令 逐个 显 式 引 
用 每 一 个 权限 。 

4. 查看 用 户 权限 

用 户 可 以 通过 查看 权限 表 中 的 适当 数据 来 查看 用 户 权 限 ， 但 随 着 表 的 增 大 ， 这 种 方法 会 变 得 越 
来 越 难以 使 用 。 所 以 通常 情况 下 ， 使 用 MySQL 提供 的 SHOW GRANTS FOR 用 户 命令 来 查询 用 户 
特定 的 权限 。 具 体 语 句 如 下 所 示 : 


mysql> SHOW GRANTS FOR root@localhost; 


执行 上 述 语句 得 到 的 结果 如 图 10-17 所 示 。 


10.1.5 “管理 用 户 


于 前 面 已 经 介绍 了 用 户 密码 的 修改 的 相关 
内 容 ， 这 里 将 主要 介绍 用 户 的 创建 、 删 除 及 重 命 
名 。 主 要 用 到 的 命令 有 : CREATE USER、DROP USER 及 RENAME USER， 下 面 将 分 别 进行 介绍 。 
1. CREATE USER 
CREATE USER 命令 用 于 创建 新 的 用 户 账户 。 在 创建 时 不 赋予 任何 权限 ， 所 以 要 想 使 用 此 用 户 
对 数据 库 及 表 进 行 某 些 操作 ， 必 须 使 用 GRANT 命令 赋予 权限 。 该 命令 的 使 用 格式 具体 如 下 所 示 : 
CREATE USER user [IDENTIFIED BY [PASSWORD] "password'] 
[,user [IDENTIFIED BY [PASSWORD] "password']].… 


现 假设 需要 创建 一 个 新 用 户 账户 YYan@localhost， 设 置 密码 为 123456， 具 体 示例 如 下 所 示 : 


mysql> CREATE USER YYan@localhost IDENTIFIED BY '123456'; 
Query OK, 0 rows affected (0.30 sec) 


图 10-17 执行 结果 
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2. DROP USER 

DROP USER 命令 用 于 创建 一 个 新 用 户 账户 。 该 命令 的 使 用 格式 具体 如 下 所 示 : 

DROP USER user [,user.]; 

在 实际 应 用 中 ， 如 果 不 再 需要 某 个 账户 ， 应 当 考 虑 将 其 删除 ， 以 确保 数据 库 及 其 数据 的 安全 。 
该 命令 将 从 权限 表 中 删除 用 户 的 所 有 信息 。 现 假设 要 删除 前 面 创建 的 账户 YYan@localhost， 具 体 示 
例如 下 所 示 : 

mysql> DROP USER YYan@localhost; 

Query OK, 0 rows affected (0.02 sec) 

3. RENAME USER 

RENAME USER 命令 用 于 重 命 名 现 有 用 户 。 该 命令 的 使 用 格式 具体 如 下 所 示 : 

RENAME USER old user TO new user 

[,old user TO new user].. 

通常 使 用 RENAME USER 命令 ， 用户 可 以 很 容易 地 重 命名 现 有 用 户 。 现 假设 将 用 户 

Jack(@localhost 重 命名 为 工 ong@localhost， 具 体 示 例如 下 所 示 : 


mysql> RENAME USER Jack@localhost TO JLong@localhost; 
Query OK, 0 rows affected (0.00 sec) 


10.1.6 ”数据 类 型 


同 其 他 数据 库 一 样 ，MySQL 也 提供 了 一 组 可 以 赋 给 表 中 各 个 列 的 数据 类 型 。 每 个 类 型 都 强制 
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数据 满足 该 数据 类 型 预先 确定 的 一 组 规则 ， 例 如 大 小 、 类 型 〈 例 如 字符 串 、 整 数 或 小 数 ) 及 格式 等 。 
本 节 主 要 介绍 MySQL 支持 的 数据 类 型 ， 提 供 关于 每 种 类 型 的 名 称 、 作 用 、 格 式 和 范围 的 信息 。 
下 面 将 MySQL 支持 的 数据 类 型 分 为 数值 、 字 符 串 及 日 期 和 时 间 三 种 类 型 来 分 别 进行 介绍 。 


1. 数值 数据 类 型 


PHP 提供 了 很 多 类 型 来 表示 数值 ， 比 如 INT、FLOAT、DOUBLE 等 都 是 在 学 习 其 他 编程 语言 
经 常 遇 到 的 。 如 表 10-2 所 示 是 PHP 提供 的 数值 数据 类 型 及 其 说 明 。 


表 10-2 ”数值 数据 类 型 


数据 类 型 说 了 明 
BIT[(M] BIT 用 于 定义 位 字段 类 型 。M 表示 每 个 值 的 位 数 ， 范 围 从 1~64。 如 果 M 被 省 略 ， 
默认 为 1 
BOOL 和 BOOLEAN BOOL 和 BOOLEAN 只 是 TINYINT(1) 的 别名 ， 该 类 型 的 变量 非 0 表示 真 。 这 里 需 
要 注意 ， 这 两 种 类 型 只 能 用 于 4.1.0 及 其 后 续 版 本 
BIGINT[(m] BIGINT 数据 类 型 提供 了 MySQL 最 大 的 整数 范围 ， 支 持 的 有 符号 数 范围 从 


INT[(M)] [UNSIGNED] 
[ZEROFILL] 
MEDIUMINT[(M)] 
[UNSIGNED] [ZEROFILL] 
INTEGER[(M] 
[UNSIGNED] [ZEROFILL] 
FLOAT[(M.D)] 
[UNSIGNED] [ZEROFILL] 


FLOAT(p) [UNSIGNED] 
[ZEROFILL] 


DECIMAL[(MLD])] 
[UNSIGNED] [ZEROFILL] 


-92233720368544775808~922372036854775807， 无 符号 数 范围 从 
0~18446744073709551615 

INT 定义 普通 大 小 的 整数 。 支 持 的 有 符号 范围 从 -2147483648~2147483647。 无 符号 
范围 是 0~4294967295 

MEDIUMINT 定义 中 等 大 小 的 整数 。 支 持 的 有 符号 范围 从 -8388608~8388607。 无 符 
号 的 范围 0~16777215 

INTEGER 是 INT 的 同义词 


FLOAT 用 于 定义 单 精度 浮 点 数 。 人 允许 的 值 是 -3.402823466E+38~ -1.175494351E-38、 
0 和 1.175494351E-38~3.402823466E+38。 这 些 是 理论 范围 ， 是 基于 IEEE 标准 的 。 
实际 的 范围 根据 硬件 或 操作 系统 的 不 同 可 能 稍微 小 些 。 

其 中 ，M 是 小 数 的 位 数 ，D 是 小 数 点 后 面 的 位 数 。 如 果 M 和 D 被 省 略 ， 根 据 硬件 
人 允许 的 限制 来 保存 值 。 单 精度 浮 点 数 精确 到 大 约 7 位 小 数位 。 如 果 指定 UNSIGNED， 
则 不 允许 负 值 。 这 里 应 该 注意 ， 使 用 浮 点 数 可 能 会 遇 到 意 想不到 的 问题 ， 因 为 在 
MySQL 中 的 所 有 计算 都 用 双 精 度 完成 

FLOAT 用 于 为 ODBC 兼容 而 提供 ， 其 中 p 表示 精度 〈 以 位 数 表示 )， 但 MySQL 只 
使 用 该 值 来 确定 是 否 结果 列 的 数据 类 型 为 FLOAT 或 DOUBLE。 如 果 p 为 从 0-~24， 
数据 类 型 变 为 没有 M 或 D 值 的 FLOAT。 如 果 p 为 从 25~53， 数 据 类 型 变 为 没有 M 
或 D 值 的 DOUBLE。 结 果 列 范围 与 单 精度 FLOAT 或 双 精度 DOUBLE 数据 类 型 
相同 

DECIMAL 用 于 定义 字符 串 浮 点 数 。M 是 小 数位 数 〈 精 度 ) 的 总 数 ，D 是 小 数 点 ( 标 
度 ) 后 面 的 位 数 。 小 数 点 和 《负数 ) 的 “一 ”符号 不 包括 在 M 中 。 如 果 D 是 0， 则 值 
没有 小 数 点 或 分 数 部 分 。DECIMAL 整数 最 大 位 数 (M) 为 65。 支 持 的 十 进 制 数 的 
最 大 位 数 (D) 是 30。 如 果 D 被 省 略 ， 默 认 是 0。 如 果 M 被 省 略 ， 默 认 是 10。 如 
果 指 定 UNSIGNED， 则 不 允许 负 值 。 这 里 需要 注意 的 是 ， 所 有 DECIMAL 列 的 基本 
计算 (+、-、*、/) 都 用 65 位 精度 完成 


DOUBLE[(M.D)] 
[UNSIGNED] [ZEROFILL] 


DOUBLE 用 于 定义 双 精 度 浮 点 数 。 支 持 的 有 符号 范围 从 -1.7976931348623157E+ 
308~ -2.2250738585072014E-308、0 和 2.2250738585072014E-308~ 
1.7976931348623157E+308。 这 些 同样 是 基于 正 EE 标准 的 理论 范围 。 实 际 的 范围 根 
据 硬件 或 操作 系统 的 不 同 可 能 稍微 小 些 。 

其 中 ，M 是 小 数 的 位 数 ，D 是 小 数 点 后 面 的 位 数 。 如 果 M 和 D 被 省 略 ， 根 据 硬件 
人 允许 的 限制 来 保存 值 . 双 精 度 浮 点 数 精确 到 大 约 15 位 小 数位 。 如 果 指定 UNSIGNED， 
则 不 允许 为 负 值 


(1) 用 户 使 用 带 符号 的 BIGINT 或 DOUBLE 类 型 值 进行 所 有 算法 ， 因 此 除了 位 函数 ， 不 应 使 用 
大 于 9223372036854775807(63 位 ) 的 无 符号 的 大 整数 。 和 否则 ， 结 果 中 的 最 后 几 位 可 能 出 错 ， 这 是 由 
于 将 BIGINT 值 转换 为 DOUBLE 进行 四 舍 五 入 时 造成 的 错误 。MySQL 可 以 在 以 下 情况 下 处 理 
BIGINT: 

。 当 使 用 整数 在 一 个 BIGINT 列 保存 大 的 无 符号 的 值 时 。 

。 在 MIN(col name) 或 MAX(col name) 中 ， 其 中 col name 指 BIGINT 列 。 

。 使 用 操作 符 (+、-、* 等 ) 并 且 两 个 操作 数 均 为 整数 时 。 

(2) 总 是 可 以 使 用 一 个 字符 串 在 BIGINT 列 中 保存 严格 整数 值 。 在 这 种 情况 下 ，MySQL 执行 字 
符 串 -数字 转换 ， 其 间 不 存在 双 精 度 表示 。 

(3) 当 两 个 操作 数 均 为 整数 值 时 ，-、+ 和 * 操 作 符 使 用 BIGINT 算法 。 这 说 明 如 果 乘 两 个 大 整数 
(或 来 自 返回 整数 的 函数 )， 当 结果 大 于 9223372036854775807 时 ， 会 得 到 意 想 不 到 的 结果 。 

2. 字符 串 数据 类 型 

PHP 提供 了 很 多 类 型 来 表示 字符 串 数据 ， 具 体 如 表 10-3 所 示 。 


表 10-3 ”字符 串 数据 类 型 


[NATIONAL] 
CHAR(M) [BINARY]| 
ASCII | UNICODE] 


CHAR 用 于 定义 固定 长 度 字符 串 ， 当 保存 时 在 右 侧 填充 空格 以 达到 指定 的 长 度 。 其 中 ，， 
M 表示 列 长 度 ， 它 的 范围 是 0~255 个 字符 。 当 检索 CHAR 值 时 尾部 空格 被 删除 。 如 果 
想 要 将 某 个 CHAR 的 长 度 设 为 大 于 255， 执 行 CREATE TABLE 或 ALTER TABLE 语句 
时 将 失败 并 提示 错误 。 

其 中 ，CHAR 是 CHARACTER 的 简写 。NATIONAL CHAR( 或 其 等 效 短 形式 NCHAR) 
是 标准 的 定义 CHAR 列 应 使 用 默认 字符 集 的 SQL 方法 。 这 在 MySQL 中 为 默认 值 ; 
BINARY 属性 是 指定 列 字符 集 的 二 元 校对 规则 的 简写 , 排序 和 比较 基于 数值 字符 值 ; 列 
类 型 CHAR BYTE 是 CHAR BINARY 的 一 个 别名 ， 这 是 为 了 保证 兼容 性 。 

MySQL 允许 创建 类 型 CHAR(0) 的 列 ， 这 主要 用 于 必须 有 一 个 列 与 不 使 用 值 的 旧版 本 的 
应 用 程序 兼容 。 当 用 户 需要 只 能 取 两 个 值 的 列 时 很 有 用 : 没有 定义 为 NOT NULL 的 一 
个 CHAR(0) 列 只 占用 一 位 ， 只 可 以 取 值 NULL 和 "( 空 字符 串 ) 


[NATIONAL] VARCHAR 用 于 定义 变 长 字符 串 ， 其 中 M 表示 最 大 列 长 度 ， 它 的 范围 是 0~65 535。 但 
VARCHAR(M) 在 实际 中 , VARCHAR 的 最 大 实际 长 度 由 最 长 的 行 的 大 小 和 使 用 的 字符 集 确定 , 最 大 有 
[BINARY] 效 长 度 是 65 532B。 


这 里 ，VARCHAR 是 字符 VARYING 的 简写 。BINARY 属性 是 指定 列 的 字符 集 的 二 元 校 
对 规则 的 简写 , 排序 和 比较 基于 数值 字符 值 . VARCHAR 保存 时 用 一 个 字 节 或 两 个 字 节 
长 的 前 缀 + 数据 表示 。 如 果 VARCHAR 列 声明 的 长 度 大 于 255， 长 度 前 缀 是 两 个 字 节 
LONGBLOB 用 于 定义 最 大 长 度 为 4294 967 295 或 4GB(23_D) 字 节 的 BLOB 列 。 
LONGBLOB 列 的 最 大 有 效 (允许 的 ) 长 度 取决 于 客户 端 /服务 器 协议 中 配置 最 大 包 大 小 
和 可 用 的 内 存 

LONGTEXT 用 于 定义 最 大 长 度 为 4294 967 295 或 4GB(22 TD) 字符 的 TEXT 列 。 
LONGTEXT 列 的 最 大 有 效 (允许 的 ) 长 度 取决 于 客户 端 /服务 器 协议 中 配置 最 大 包 大 小 
和 可 用 的 内 存 


LONGBLOB 


LONGTEXT 


MEDIUMBLOB MEDIUMBLOB 用 于 定义 最 大 长 度 为 16 777 215(2*-1) 字 节 的 BLOB 列 
MEDIUMTEXT MEDIUMTEXT 用 于 定义 最 大 长 度 为 16 777 215C22-_1) 字 符 的 TEXT 列 
BINARY(M) BINARY 类 型 类 似 于 CHAR 类 型 ， 用 于 保存 二 进 制 字 节 字 符 串 而 不 是 非 二 进 制 字符 串 


VARBINARY(M) VARBINARY 类 型 类 似 于 VARCHAR 类 型 ， 用 于 保存 二 进 制 字 节 字符 串 而 不 是 非 二 进 
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续 表 


数据 类 型 说 了 明 
BLOB[(M)] BLOB 用 于 定义 最 大 长 度 为 65 535(2'1) 字 节 的 BLOB 列 。 用 户 可 以 给 出 该 类 型 的 可 选 长 
度 M。 如 果 给 出 ， 则 MySQL 将 列 创建 为 最 小 的 但 足以 容纳 M 字 节 长 的 值 的 BLOB 类 型 
TINYBLOB TINYBLOB 用 于 最 大 长 度 为 255(2*_1) 字 节 的 BLOB 列 
TINYTEXT TINYTEXT 用 于 定义 最 大 长 度 为 255(2_1) 字 符 的 TEXT 列 
TEXT[(MW] TEXT 用 于 定义 最 大 长 度 为 65 535(2' 1) 字符 的 TEXT 列 。 用 户 可 以 给 出 可 选 长 度 M。 如 
果 给 出 ， 则 MySQL 将 列 创建 为 最 小 的 但 足以 容纳 M 字符 长 的 值 的 TEXT 类 型 
ENUM('valuel'. ENUM 用 于 定义 枚 举 类 型 。 只 能 有 一 个 值 的 字符 串 ， 从 值 列 valuel、value2、…，NULL 中 
‘value2',...) 或 特殊 的 错误 值 中 选 出 。ENUM 列 最 多 可 以 有 65 535 个 截然 不 同 的 值 。ENUM 值 在 内 部 用 
整数 表示 。 如 果 声 明了 NOT NULL， 则 列表 的 第 一 个 成 员 是 默认 值 
SET(valuel, SET 为 一 组 预定 义 值 中 的 零 个 或 多 个 值 提供 了 一 种 方法 。 例 如 ， 字 符 串 对 象 可 以 有 和 零 个 或 
并 ‘value2',...) 多 个 值 ， 每 个 值 必须 来 自 列 值 valuel1，value2，SET 列 最 多 可 以 有 64 个 成 员 。SET 值 在 内 
3 部 用 整数 表示 
名 
2 3. 日 期 和 时 间 数 据 类 型 
到 同样 ，PHP 也 提供 了 很 多 类 型 来 表示 日 期 和 时 间 数 据 ， 有 具体 如 表 10-4 所 示 。 
学 表 10-4 “日 期 和 时 间 数 据 类 型 
习 数据 类 型 说 阴 
i DATE DATE 用 于 定义 日 期 类 型 的 列 。 它 支持 的 范围 从 1000-01-01~9999-12-31。MySQL 以 YYYY- 
MM-DD 格式 显示 DATE 值 ， 但 允许 使 用 字符 串 或 数字 为 DATE 列 分 配 值 
TIME TIME 用 于 定义 时 间 类 型 的 列 。 它 支持 的 范围 从 -838:59:59~838:59:59。MySQL 以 HH:MM:SS 
格式 显示 TIME 值 ， 但 允许 使 用 字符 串 或 数字 为 TIME 列 分 配 值 
DATETIME DATETIME 是 日 期 和 时 间 的 组 合 。 它 支持 的 范围 是 1000-01-01 00:00:00~9999-12-31 
23:59:59。MySQL 以 YYYY-MM-DD HH:MM:SS 格式 显示 DATETIME 值 , 但 允许 使 用 字符 
串 或 数字 为 DATETIME 列 分 配 值 
YEAR[CI4)] YEAR 用 于 定义 2 位 或 4 位 格式 的 年 ,默认 是 4 位 格式 .在 4 位 格式 中 , 允许 的 值 是 1901~2155 


和 0000。 在 两 位 格式 中 ， 人 允许 的 值 是 70~69， 表 示 从 1970 年 ~2069 年 。MySQL 以 YYYY 
格式 显示 YEAR 值 ， 但 允许 使 用 字符 串 或 数字 为 YEAR 列 分 配 值 

TIMESTAMP[(M)] | TIMESTAMP 用 于 定义 时 间 戳 类 型 的 列 。 它 支持 的 范围 从 1970-01-01 00:00:00~2037 年 。 
TIMESTAMP 列 用 于 INSERT 或 UPDATE 操作 时 记录 日 期 和 时 间 。 如 果 用 户 不 分 配 一 个 值 ， 
表 中 的 第 一 个 TIMESTAMP 列 自动 设置 为 最 近 操 作 的 日 期 和 时 间 。 也 可 以 通过 分 配 一 个 
NULL 值 ， 将 TIMESTAMP 列 设置 为 当前 的 日 期 和 时 间 。 
TIMESTAMP 值 返回 后 显示 为 YYYY-MM-DD HH:MM:SS 格式 的 字符 串 ， 显 示 宽 度 固 定 为 
19 个 字符 。 如 果 要 获得 数字 值 ， 则 应 在 TIMESTAMP 列 添加 +0 


10.1.7 ”管理 数据 库 


在 进行 PHP 编程 以 前 ， 对 MySQL 数据 库 进 行 适当 的 管理 也 是 非常 必要 的 ， 所 以 本 节 将 介绍 如 
何 创建 、 查 看 、 使 用 和 删除 MySQ 数据 库 。 

1. 创建 数据 库 

创建 MySQL 数据 库 主要 有 两 种 方法 ， 最 简单 的 方法 是 在 MySQL 客户 端 使 用 CREATE 
DATABASE 命令 进行 创建 。 如 下 所 示 用 于 创建 一 个 名 为 guestbook 的 数据 库 : 
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mysql> CREATE DATABASE guestbook; 
Query OK, 1 row affected (0.23 sec) 


另 一 种 方法 是 通过 mysqladmin 客户 端 创建 数据 库 ， 代 码 如 下 所 示 : 


%> mysqladmin -u root -p create guestbook 


eseseee 


Enter password: 
$> 


如 果 创 建 数据 库 失 败 ， 原 因 可 能 是 权限 不 够 或 不 正确 ， 或 者 尝试 创建 已 经 存在 的 数据 库 。 

2. 查看 数据 库 

前 面 创 建 了 数据 库 guestbook， 可 以 通过 执行 SHOW DATABASES 命令 来 获取 服务 器 上 的 数据 
库 列 表 ， 以 查看 已 经 存在 的 数据 库 ， 具 体 示例 如 图 10-18 所 示 。 

实现 上 述 功能 ， 用 户 也 可 以 使 用 mysqlshow 命令 得 到 此 数据 库 列表 。 用 户 只 能 看 到 自己 拥有 某 
些 权限 的 数据 库 ， 除 非 用 户 拥有 全 局 SHOW 
DATABASES 权限 。 

如 果 服 务 器 以 --skip-show-database 选项 为 
起 始 ， 除 非 用 户 拥有 SHOW DATABASES 权 
限 ， 否 则 用 户 不 能 使 用 该 命令 。 另 外 ， 用 户 也 
可 以 考虑 使 用 SHOW SCHEMAS 来 得 到 数据 
库 列表 。 

3. 使 用 数据 库 
数据 库 创 建 以 后 , 可 以 通过 “使 用 ”(use) 二 
数据 库 ， 将 其 指定 为 默认 的 工作 数据 库 ， 这 样 10-18 查看 数据 库 

后 面 对 数 据 库 的 操作 都 是 基于 该 数据 库 的 。 现 
假设 要 使 用 前 面 创建 的 数据 库 guestbook， 有 具体 示例 如 下 所 示 : 
mysql> USE guestbook; 

Database changed 


用 户 还 可 以 通过 mysql 客户 端 登录 时 ， 在 命令 行 传 入 数据 库 名 ， 直 接 切换 到 该 数据 库 。 具 体 代 
码 如 下 所 示 : 
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%>mysql -u root -p guestbook; 


4. 删除 数据 库 

当 某 些 数据 库 不 再 使 用 时 可 以 将 其 删除 ， 用 户 可 以 通过 在 MySQL 客户 端 使 用 DROP 命令 实现 
该 功能 。 它 的 使 用 格式 如 下 所 示 : 

DROP {DATABASE | SCHEMA} [IF EXISTS] db_ name; 

该 命令 的 具体 示例 如 下 所 示 : 


mysql> DROP DATABASE guestbook; 
Query OK, 0 rows affected (0.00 sec) 
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如 果 要 使 用 DROP DATABASE， 用 户 需 要 获得 数据 库 DROP 权限 。 用 户 可 以 使 用 正 EXISTS 
来 防止 当 数据 库 不 存在 时 发 生 错误 ， 也 可 以 使 用 DROP SCHEMA 完成 DROP DATABASE 的 功能 。 

如 果 用 户 对 一 个 带 有 符号 连接 的 数据 库 使 用 DROP DATABASE 命令 ， 则 连接 和 原 数据 库 都 被 
取消 。DROP DATABASE 会 返回 已 被 取消 的 表 的 数目 。 在 正常 操作 中 ，MySQL 自身 会 创建 出 一 些 
文件 和 目录 。DROP DATABASE 语句 会 从 给 定 的 数据 库 目录 中 取消 这 些 文件 和 目录 。 


10.1.8 ”管理 表 


在 介绍 了 管理 数据 库 的 相关 内 容 后 ， 本 节 将 介绍 如 何 创建 、 复 制 、 查 看 和 删除 表 ， 以 及 查看 、 

1. 创建 表 

在 创建 了 数据 库 以 后 ， 就 可 以 使 用 CREATE TABLE 语句 来 创建 表 了 。 该 语句 的 具体 格式 如 下 
所 示 : 

CREATE [TEMPORARY] TABLE [IF NOT EXISTS] tb] name 


[(create definition,...)] 
[table options] [select statement] 


上 述 语句 很 简单 ,其实 CREATE TABLE 有 很 多 选项 和 子 句 , 读者 可 以 查看 MySQL 参考 手册 来 
获得 详细 的 使 用 格式 。 下 面 通过 在 数据 库 guestbook 中 创建 一 个 存储 用 户 留 言 的 表 message， 来 说 明 
该 语句 是 如 何 使 用 的 。 具 体 示例 如 下 所 示 : 

mysql> CREATE TABLE message( 

-> msgid tinyint UNSIGNED NOT NULL AUTO INCREMENT, 
-> username varchar(20), 
-> email varchar (50), 
-> title varchar (50), 
-> content varchar (1000), 
-> time datetime, 
-> primary key (msgid)); 
Query OK, 0 rows affected (0.31 sec) 


在 执行 上 述 语句 之 前 ， 必 须 通过 USE 命令 使 用 guestbook 数据 库 ， 或 者 使 用 如 下 所 示 的 方式 来 
指定 当前 要 使 用 的 数据 库 : 


database name.table name 


在 默认 情况 下 ， 如 果 用 户 试图 创建 一 个 已 经 存在 的 表 ，MySQL 会 产生 一 个 错误 。 为 了 避免 产 
生 这 种 错误 , CREATE TABLE 语句 提供 了 一 个 子 句 ,如 果 用 户 希 望 在 目标 表 已 经 存在 的 情况 下 退出 
表 的 创建 ， 就 可 以 包含 这 个 子 句 。 这 里 的 子 句 是 正 NOT EXISTS， 它 的 具体 使 用 如 下 所 示 : 
mysql> CREATE TABLE IF NOT EXISTS message( 
-> msgid tinyint UNSIGNED NOT NULL AUTO INCREMENT, 


-> username varchar(20), 
-> email varchar (50), 


-> Primary key (msgid));; 


这 里 需要 注意 的 是 , 上述 语句 中 的 表 是 否 创建 , 都 会 在 返回 到 MySQL 命令 提示 符 时 显示 “Query 
OK” 消 息 。 

此 外 , 还 可 以 通过 CREATE TABLE 语句 来 创建 临时 表 , 只 要 加 入 关键 字 TEMPORARY 就 行 了 ， 
具体 示例 如 下 所 示 : 


CREATE TEMPORARY TABLE msgtemp ...; 


2. 复制 表 

复制 表 是 基于 现 有 的 表 来 创建 一 个 新 表 。 下 面 语句 将 得 到 message 表 的 一 个 完全 副本 ， 这 个 副 
本 表 为 msg。 具 体 示例 如 下 所 示 : 

mysql> CREATE TABLE msg SELECT * FROM message; 


Query OK, 0 rows affected (0.19 sec) 
Records: 0 Duplicates: 0 Warnings: 0 


成 功 执行 上 述 语 句 将 向 数据 库 中 增加 一 个 和 message 结构 相同 的 表 ， 但 这 种 情况 并 不 多 见 ， 在 
实际 应 用 中 经 常 只 基于 现 有 表 的 几 个 列 来 创建 一 个 表 。 那么 通过 在 CREATE SELECT 语句 中 指定 列 
就 可 以 了 。 具 体 如 下 所 示 : 

mysql> CREATE TABLE msg2 SELECT msgid,title,content FROM message; 


Query OK, 0 rows affected (0.11 sec) 
Records: 0 Duplicates: 0 Warnings: 0 


3. 查看 表 

可 以 通过 执行 SHOW TABLES 命令 来 
获取 当前 数据 库 中 表 的 情况 ， 具 体 示例 如 图 
10-19 所 示 。 

4. 删除 表 

当 某 些 数据 库 不 再 使 用 时 可 以 将 其 删 
除 ， 对 表 来 说 也 是 如 此 。 用 户 可 以 通过 在 
MySQL 客户 端 使 用 DROP 命令 实现 该 功能 。 它 的 使 用 格式 如 下 所 示 : 

DROP [TEMPORRRY] TABLE [IF EXISTS] 


tbl name [，tbl name] ... 
[RESTRICT | CASCADE] 


现 假设 要 删除 前 面 创建 的 表 msg， 具 体 如 下 所 示 : 


mysql> DROP TABLE msg; 
Query OK, 0 rows affected (0.09 sec) 


其 实 ， 可 以 使 用 该 语句 同时 删除 多 个 表 ， 具 体 代 码 如 下 所 示 : 


mysql> DROP TABLE msg,msg2; 
Query OK, 0 rows affected (0.08 sec) 


图 10-19 查看 数据 库 中 的 表 


已 站 


章 
加 
CO 
名 
全 
数 
据 
jE 


卫 
2 
五 
十 
三 

AR 
ea 
QO 
i 
| 一 
TU 
全 
学 
习 
手 
册 


5. 查看 表 结构 

如 果 用 户 想 要 知道 一 个 表 的 结构 ， 可 以 
使 用 DESCRIBE 命令 , 它 用 于 显示 表 中 每 个 
列 的 信息 。 现 假设 要 查看 表 message 的 结构 ， 
具体 示例 如 图 10-20 所 示 。 

在 图 10-20 中 ，Field 显示 列 名 字 ，Type 
是 列 的 数据 类 型 ，Null 表示 列 是 否 能 包含 
NULL 值 ，Key 显示 列 是 否 被 索引 ，Default 
指定 列 的 默认 值 。 如 果 表 有 索引 ，SHOW INDEX FROM tbl name 则 生成 有 关 索 引 的 信息 。 

6. 修改 表 结构 

在 实际 开发 中 ， 经 常会 发 现 已 经 存在 的 某 些 表 已 经 不 符合 要 求 ， 这 时 就 需要 对 表 的 结构 进行 修 
改 。 在 MySQL 中 ， 修 改 表 结构 使 用 ALTER 语句 ， 它 的 部 分 语法 结构 具体 如 下 所 示 : 

ALTER [IGNORE] TABLE tbl name 

alter specification [, alter specification] ... 

该 语句 同 CREATE TABLE 语句 一 样 也 很 复杂 ,具有 许多 子 句 ,读者 可 以 查看 MySQL 参考 手册 。 
下 面 通过 一 些 示 例 来 说 明 ALTER TABLE 的 用 法 。 

(1) 现 假 设 在 前 面 创建 的 message 表 中 增加 一 列 ， 该 列 为 url 用 于 存储 用 户 的 个 人 主页 网 址 。 
具体 语句 如 下 所 示 : 

mysql> ALTER TABLE message ADD COLUMN url varchar(50); 


Query OK, 0 rows affected (0.31 sec) 
Records: 0 Duplicates: 0 Warnings: 0 


上 述 语句 成 功 执行 后 ， 使 用 DESCRIBE 命令 查看 表 message 的 结构 ， 将 会 看 到 新 的 列 url 放 在 
了 表 的 最 后 位 置 。 

(2) 在 增加 新 的 列 时 ， 可 以 使 用 适当 的 关键 字 (如 FIRST、AFTER 和 LAST) 来 控制 新 列 的 位 
置 。 现 假设 在 username 列 的 后 面 增加 一 列 ， 该 列 为 phone 用 于 存储 用 户 的 联系 方式 。 具体 语 句 如 下 
所 示 : 

mysql> ALTER TABLE message ADD COLUMN phone varchar (12) AFTER username; 


Query OK, 0 rows affected (0.19 sec) 
Records: 0 Duplicates: 0 Warnings: 0 


(3) 给 已 经 存在 的 列 添加 NOT NULL 约束 。 具 体 代 码 如 下 所 示 : 


图 10-20 查看 表 message 的 结构 


mysql> ALTER TABLE message CHANGE phone phone varchar(12) NOT NULL; 

Query OK, 0 rows affected (0.17 sec) 

Records: 0 Duplicates: 0 Warnings: 0 

这 里 要 注意 的 是 ， 执 行 该 语句 要 提供 旧 的 和 新 的 列 名 称 ， 即 使 旧 的 和 新 的 列 名 称 是 一 样 的 也 是 
如 此 。 

(4) 当 表 中 的 某 些 列 不 再 需要 时 ， 可 以 删除 该 列 。 具 体 代码 如 下 所 示 : 


mysql> ALTER TABLE message DROP phone; 


Query OK, 0 rows affected (0.20 sec) 
Records: 0 Duplicates: 0 Warnings: 0 


10.1.9 创建 索引 


索引 是 数据 库 中 一 种 常用 且 重 要 的 数据 库 对 象 ， 它 类 似 于 图 书 中 的 目录 。 在 图 书 中 ， 目 录 人 允许 
用 户 不 必 翻 阅 整 本 图 书 就 能 根据 目录 中 的 页 数 迅速 找到 所 需 内 容 。 在 数据 库 中 ， 索 引 也 允许 数据 库 
应 用 程序 迅速 找到 表 中 特定 的 数据 ， 而 不 必 扫 描 整 个 数据 库 中 的 数据 。 在 图 书 中 ， 目 录 是 内 容 和 相 
点 页 码 的 列表 。 在 数据 库 中 ， 索 引 是 表 中 数据 和 相应 存储 位 置 的 列表 。 使 用 索引 可 以 大 大 减少 用 于 
查找 指定 数据 的 时 间 ， 改 善 数据 库 性 能 。 在 MySQL 中 索引 可 分 为 主键 索引 、 唯 一 索引 、 常 规 索引 
和 全 文 索引 。 
1. 主键 索引 
主键 索引 是 关系 数据 库 中 最 常见 的 索引 类 型 。 它 通过 主键 自身 的 唯一 性 来 标识 每 条 记录 。 因此 ， 
该 键 必 须 是 该 记录 所 表示 实体 唯一 拥有 的 值 ， 或 者 是 数据 库 自 动 生成 的 唯一 值 ， 比 如 ， 通 过 
AUTO _INCREMENT 约束 的 列 值 。 该 唯一 值 确定 了 每 条 记录 都 有 唯一 的 主键 索引 。 现 假设 存在 一 个 
表 reply， 它 用 于 回复 前 面 创建 的 message 表 中 的 用 户 留言 。 创 建 此 表 的 具体 代码 如 下 所 示 : 
mysql> CREATE TABLE replyl( 
-> repid tinyint(3) unsigned not null auto increment, 
-> msgid tinyint(3) unsigned not null, 
-> recontent varchar(1000), 
-> retime datetime, 
-> primary key (repid)); 
Query OK, 0 rows affected (0.13 sec) 


在 上 述 语句 中 ，repid 字段 在 每 次 插入 记录 时 自动 加 1 (从 1 开始 )， 所 以 reply 表 不 会 包括 完全 
相同 的 记录 。 下 面 代码 用 于 向 表 中 插入 三 条 记录 ， 具 体 如 下 所 示 : 

mysql> INSERT INTO reply VALUES (null,]1,'Hello PHP!',"'2007-06-26 18:07:11°'); 

mysql> INSERT INTO reply VALUES (null,2, 'http://www.itzcn.net','2007-06-26 

QO 

mysql> INSERT INTO reply VALUES (null,3,'http://www.itying.net','2007-06-26 

LDeOTE LL 

成 功 执行 上 述 语句 后 ， 使 用 SELECT 语句 查看 表 中 记录 ， 具 体 示 例如 图 10-21 所 示 。 

从 上 面 的 示例 可 以 看 出 , 字段 repid 在 每 
次 插入 记录 时 自动 加 1， 这 就 确保 了 在 reply 
表 中 记录 的 唯一 性 。 

2. 唯一 索引 

唯一 索引 用 于 防止 创建 重复 的 值 。 与 主 
键 索引 的 不 同 之 处 在 于 ， 每 个 表 只 有 一 个 主 
键 索引 ， 但 可 以 有 多 个 唯一 索引 。 例 如 回复 


用 户 的 留言 ， 在 通常 情况 下 ， 每 个 留言 的 回 
复 记 录 只 有 一 条 ， 再 次 回复 时 修改 这 条 记录 图 10-21 查看 表 中 记录 
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的 回复 内 容 就 行 了 , 所 以 如 果 在 reply 表 中 引用 了 两 次 或 多 次 用 户 留言 D Cmsgid) 就 不 是 很 理想 了 。 
如 下 语句 在 创建 表 时 指定 msgid 字段 唯一 索引 ， 有 具体 如 下 所 示 : 


mysql> CREATE TABLE reply( 
-> repid tinyint(3) unsigned not null auto increment, 
-> msgid tinyint(3) unsigned not null UNIQUE, 
-> recontent varchar(1000), 
-> retime datetime, 
-> primary key (repid)); 


上 述 语句 中 只 指定 一 个 唯一 索引 ， 在 实际 应 用 中 可 指定 多 个 ， 在 需要 时 还 可 以 指定 多 个 字段 唯 
一 索引 。 现 假设 可 以 允许 用 户 插入 重复 的 msgid 值 ， 也 可 出 现 重复 的 recontent 值 ， 但 不 允许 出 现 重 
复 msgid 和 recontent 的 组 合 。 为 了 实现 该 功能 , 用 户 可 以 通过 创建 多 字段 唯一 索引 来 强制 这 种 约束 。 
修改 后 的 创建 表 reply 的 语句 如 下 所 示 : 


mysql> CREATE TABLE reply( 
-> repid tinyint(3) unsigned not null auto increment, 
-> msgid tinyint(3) unsigned not null, 
-> recontent varchar(1000), 
->UNIQUE (msgid, recontent), 
-> retime datetime, 
-> Primary key (repid)); 


对 于 上 述 创建 的 表 ， 表 中 记录 类 似 于 如 下 所 示 : 


www.itzcn.com 2007-06-26 18:20:12 
Wwww.itzcn.net 2007-06-26 19:30:12 
Www.itzcn.com 2007-06-26 20:20:12 
www.itying.net 2007-06-26 22:20:12 


3. 常规 索引 

常规 索引 主要 用 于 在 非 主键 甚至 并 非 唯一 的 字段 上 搜索 并 进行 优化 ， 这 样 就 可 以 为 这 些 字段 建 
立 索 引 来 优化 这 种 搜索 。 该 种 索引 也 称 为 正常 索引 。 

(1) 单字 段 常 规 索引 

如 果 表 中 的 某 个 字段 将 成 为 大 量 选择 查询 的 焦点 ， 就 应 当 使 用 单字 段 常规 索引 。 例 如 ， 前 面 创 
建 的 存储 留言 信息 的 表 message 包括 6 个 字段 : 唯一 标签 DD、 用 户 名 、 电 子 信箱 、 留 言 标题 、 留 言 
内 容 、 留 言 时 间 。 在 通常 情况 下 搜索 都 要 针对 用 户 名 或 者 电子 邮件 进行 ， 所 以 要 为 用 户 名 创建 一 个 
常规 索引 ， 为 电子 邮件 地 址 创建 唯一 索引 。 具 体 如 下 所 示 : 


mysql> CREATE TABLE message( 
-> msgid tinyint UNSIGNED NOT NULL AUTO INCREMENT, 
-> username varchar (20), 
-> email varchar (50) UNIQUE, 
-> title varchar (50) ， 
-> content varchar(1000), 
-> time datetime, 
—>INDEX (username), 
-> primary key (msgid)); 


ON 
w mm FF 


(2) 多 字段 常规 索引 

如 果 某 些 字段 经 常 在 获取 查询 中 一 起 使 用 ， 则 可 以 使 用 多 字段 索引 。MySQL 的 多 字段 索引 方 
法 基于 一 种 称 为 最 左前 级 (leftmost prefixing) 的 策略 。 最 左前 绥 指 出 包含 字段 A、B 和 C 的 任何 多 
字段 可 以 提高 涉及 如 下 字段 组 合 的 查询 的 性 能 : 

。 A。 

。 A、B. 

。A、B、C。 

下 面 的 示例 用 于 创建 多 个 字段 常规 索引 ， 具 体 如 下 所 示 : 


mysql> CREATE TABLE message( 
-> msgid tinyint UNSIGNED NOT NULL AUTO INCREMENT, 
-> username varchar(20), 
-> email varchar (50) UNIQUE, 
-> title varchar (50), 
-> content varchar(1000), 
-> time datetime, 
->INDEX (username, title, content), 
-> primary key (msgid)); 


上 述 语 句 用 于 创建 三 个 索引 。 第 一 个 索引 是 msgid 的 主键 索引 ; 第 二 个 是 email 的 唯一 索引 ; 
第 三 个 是 多 字段 常规 索引 ， 它 包括 三 个 字段 username、title 和 content， 所 以 只 要 查询 涉及 以 下 任何 
字段 组 合 都 将 提高 搜索 的 速度 : 

e USsername、title、content。 

e Uusermame、 title。 

@ USeIname。 

4. 全 文 索引 

从 版 本 3.23.23 开始 ，MySQL 开始 支持 全 文 索引 。 全文 索 引 在 MySQL 中 是 一 个 FULLTEXT 类 
型 索引 。FULLTEXT 索引 用 于 MyISAM 表 ， 可 以 在 CREATE TABLE 时 或 之 后 使 用 ALTER TABLE 
或 CREATE INDEX 在 CHAR、VARCHAR 或 TEXT 列 上 创建 。 对 于 大 的 数据 库 ， 将 数据 装载 到 一 
个 没有 FULLTEXT 索引 的 表 中 ,然后 再 使 用 ALTER TABLE( 或 CREATE INDEX) 创 建 索 引 ， 这 将 会 
非常 快 。 将 数据 装载 到 一 个 已 经 有 FULLTEXT 索引 的 表 中 ， 将 会 非常 慢 。 

创建 全 文 索引 与 创建 其 他 类 型 的 索引 很 相似 。 这 里 再 次 修改 前 面 创建 表 message 的 语句 ， 为 
content 字段 增加 全 文 索引 。 具 体 如 下 所 示 : 


mysql> CREATE TABLE message( 
-> msgid tinyint UNSIGNED NOT NULL AUTO INCREMENT, 
-> username varchar(20), 
-> email varchar (50), 
-> title varchar (50), 
-> content varchar(1000), 
-> time datetime, 
->FULLTEXT (content), 
-> primary key (msgid)); 
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在 上 述 语句 中 ,除了 典型 的 主键 索引 ,还 创建 了 一 个 包括 content 字段 的 全 文 索 引 。 从 这 里 可 以 
看 出 它们 的 创建 非常 相似 ， 但 基于 全 文 索引 的 获取 查询 却 有 所 不 同 。 基 于 全 文 索引 获取 数据 时 ， 
SELECT 查询 使 用 两 个 特殊 的 MySQL 函数 ，MATCHO 和 AGAINSTO。 

函数 MATCHO 对 照 一 个 文本 集 (包含 在 一 个 FULLTEXT 索引 中 的 一 个 或 多 个 列 的 列 集 ) 执行 
一 个 自然 语言 搜索 一 个 字符 串 。 搜 索 字符 串 作 为 AGAINSTO 函 数 的 参数 被 给 定 。 搜 索 以 忽略 字母 大 
小 写 的 方式 执行 。 对 于 表 中 的 每 个 记录 行 ，MATCHO 函 数 返回 一 个 相关 性 值 。 即 ， 在 搜索 字符 串 与 
记录 行 在 MATCHO 函 数列 表 中 指定 的 列 的 文本 之 间 的 相似 性 尺度 。 现在 假设 message 表 中 有 以 下 记 
录 ， 如 表 10-5 所 示 。 

表 10-5 message 表 中 的 记录 


msgid | username email title content time 
1 宋 岩 岩 syy@itying net | YYan'sBlog | My Blog is http://yyan.itying .net | 2007-06-26 10:07:11 


2 ljh@itying.net | Jack’sSpace | My Space is http://www.itying.net | 2007-06-27 16:17:16 
3 CodisBBS | My BBSishttp://www.codicn |2007-06-28 12:37:16 
4 ITzCN'sBBS | My BBS is http://bbsiitzcn.com | 2007-06-28 16:26:16 


下 面 使 用 MATCHO 和 AGAINSTO 函 数 来 针对 全 文 索引 content 字段 执行 自然 搜索 。 具体 语句 如 
下 所 示 : 


SELECT username,email,title FROM message WHERE MATCH (content) AGAINST('BBS'); 


执行 上 述 语 句 得 出 的 结果 类 似 于 如 下 所 示 : 


+------------ 一 二 -一 一 一 一 一 一 一 一 一 一 一 一 一 一 二 -一 一 一 一 一 一 一 一 一 一 一 一 一 一 + 
| username | email Wee 1 
+------------- +--------------- 一 二 -一 一 一 一 一 一 一 一 一 一 一 一 一 一 + 
| 李 晓 渊 | lxy@itzcn.com | ItzCN'sBBS 1 
1 赵 星 | zc@cody.cn | Codi'sBBS 1 
+------------ 一 +---------------- +--------------- 十 


当 MATCHO 函 数 被 使 用 在 一 个 WHERE 子 句 中 时 ,返回 的 记录 行 被 自动 地 以 相关 性 从 高 到 低 的 
次 序 排序 。 相 关 性 值 是 非 负 的 浮 点 数字 。 零 相关 性 意味 着 不 相似 。 相 关 性 的 计算 是 基于 词 在 记录 行 
中 的 数目 、 在 行 中 唯一 词 的 数目 、 在 集中 词 的 全 部 数目 和 包含 一 个 特殊 词 的 文档 (记录 行 ) 的 数目 。 

前 面 的 例子 是 函数 MATCHO 使 用 上 的 一 些 基本 说 明 。 记录 行 以 相关 性 递减 的 顺序 返回 。 下 面 示 
例 用 于 显示 如 何 检索 一 个 明确 的 相关 性 值 。 如 果 即 没有 WHERE 也 没有 ORDER BY 子 句 ， 返 回 行 
是 不 排序 的 。 具 体 示例 如 下 所 示 : 


SELECT MATCH (content) AGAINST('BBS') FROM message; 


执行 时 ，MySQL 会 搜索 表 中 的 第 一 条 记录 ， 并 计算 各 条 记录 的 相关 值 。 结 果 类 型 如 下 所 示 : 


二- 一 一 一 一 一 一 一 一 一 一 一 一 一 一 一 一 一 一 一 一 一 一 一 一 一 一 一 一 一 一 一 一 一 一 一 一 一 一 一 一 一 一 一 一 一 一 一 一 一 一 + 
1 match (description) against("Rpache') [| 
二 一 一 一 一 一 一 一 一 一 一 一 一 一 一 一 一 一 一 一 一 一 一 一 一 一 一 一 一 一 一 一 一 一 一 一 一 一 一 一 一 一 一 一 一 一 一 一 一 一 一 一 + 
上 0 | 
上 0 ] 
上 1 


0.32658945238712 


1 0.56658945590782 1 


10.1.10 备份 数据 库 


备份 数据 库 是 指 把 数据 库 复 制 到 转 储 设备 ， 也 就 是 创建 一 个 当前 数据 库 的 副本 。 其 中 ， 转 储 设 
备 是 指 用 于 放置 数据 库 备 份 的 磁带 或 磁盘 。 通 常 也 将 存放 于 转 储 设备 中 的 数据 库 的 备份 称 为 原 数据 
库 的 副本 或 转 储 。 

备份 数据 库 可 以 在 数据 库 表 丢失 或 损坏 ， 或 者 发 生 系统 崩溃 的 情况 下 ， 用 户 能 够 将 数据 库 表 尽 
可 能 地 恢复 到 裔 让 发 生 时 的 状态 。 备 份 数据 库 的 两 个 主要 方法 是 用 mysqldump 程序 或 直接 备份 数据 
库 文件 〈 如 用 cp、cpio 或 tar 等 )。 每 种 方法 都 有 其 优 缺 点 : 

(1) 直接 备份 方法 在 服务 器 外 部 进行 ， 并 且 用 户 必 须 采 取 措 施 保证 没有 客户 正在 修改 将 要 复制 
的 表 。 如 果 用 户 想 用 文件 系统 备份 来 备份 数据 库 ， 也 会 发 生 同样 的 问题 : 如 果 数 据 库 表 在 文件 系统 
备份 过 程 中 被 修改 ， 进 入 备份 的 表 文件 数据 不 一 致 ， 那 么 对 以 后 的 恢复 表 将 失去 意义 。 文 件 系 统 备 
份 与 直接 备份 文件 的 区 别 是 对 后 者 用 户 完全 控制 了 备份 过 程 ， 这 样 能 采取 措施 确保 服务 器 让 表 不 受 
干扰 。 

(2) mysqldump 比 直接 复制 要 慢 。mysqldump 生成 能 够 移植 到 其 他 计算 机 的 文本 文件 ， 甚 至 那 
些 有 不 同 硬件 结构 的 计算 机 上 。 直 接 备份 文件 不 能 移植 到 其 他 计算 机 上 ， 除 非 正在 备份 的 表 使 用 
MyISAM 存储 格式 。ISAM 表 只 能 在 相似 的 硬件 结构 的 计算 机 上 备份 。 在 MySQL 3.23 中 引入 的 
MyISAM 表 存 储 格 式 解决 了 该 问题 ， 因 为 该 格式 是 与 计算 机 无 关 的 ， 所 以 直接 备份 文件 可 以 移植 到 
具有 不 同 硬件 结构 的 计算 机 上 。 但 需要 满足 两 个 条 件 : 另 一 台 计算 机 必须 也 运行 MySQL 3.23 或 以 
后 版 本 ， 而 且 文件 必须 以 MyISAM 格式 表示 ， 而 不 是 ISAM 格式 。 

1. 使 用 mysqldump 备份 数据 库 

mysqldump 用 于 备份 MySQL 服务 器 中 现 有 的 表 数 据 或 表 结构 。 用 户 使 用 mysqldump 程序 产生 
数据 库 备 份 文件 时 ， 默 认 情况 下 ， 文 件 内 容 包含 创建 表 的 CREATE 语句 和 表 中 行 数据 的 INSERT 语 
句 。 换 句 话 说，mysqldump 产生 的 输出 可 在 以 后 用 作 mysql 的 输入 来 重建 数据 库 。 用 户 可 以 指定 备 
份 服务 器 中 的 某 一 个 、 部 分 或 所 有 数据 库 ， 甚 至 可 以 只 是 给 定数 据 库 中 的 某 个 表 。 调 用 mysqldump 
可 以 使 用 如 下 所 示 的 三 种 不 同 语法 : 

%>mysqldump [options] database [tables] 


%>mysqldump [options] -databases [options] databasel [database2...] 
%>mysqldump [options] -all-databases [options] 


下 面 通过 一 些 示 例 来 介绍 mysqldump 的 使 用 。 现 假设 要 备份 本 地 服务 器 中 所 有 数据 库 的 表 结 构 
到 文件 mysql-table-structures.sql， 具 体 示例 如 下 所 示 : 


%>mysqldump -u root -all-databases -no-data > mysql-table-structures.sql 


上 述 语 句 输出 定向 到 一 个 文件 ， 如 果 省 略 此 文件 ， 将 输出 到 标准 输出 ， 即 屏幕 布 上 。 如 下 所 示 
语句 用 于 备份 数据 库 guestbook 的 数据 : 


%>mysqldump -u root -p -no-create-info guestbook > mysql-guestbook-data.sql 


己 小 


章 
有 
< 
(dp) 
名 
a 
数 
据 
库 


SO 
Sl 
卫 
十 

三 
< 

op) 
名 
Fe 
完 
宇 
学 
习 
手 
册 


下 面 的 语句 用 于 备份 guestbook 数据 库 中 两 个 表 (message 和 reply) 的 结构 和 数据 ， 在 每 个 
CREATE 语句 前 包括 了 DROP TABLE 语句 。 当 需要 重复 地 重新 创建 存在 的 数据 表 时 ， 这 样 做 很 有 
用 ， 因 为 尝试 创建 已 经 存在 的 表 将 导致 错误 ， 所 以 在 创建 之 前 先 删除 已 经 存在 并 与 之 同名 的 表 。 有 具 
体 示 例如 下 所 示 : 


%>mysqldump -u root -p -add-drop-table message reply staff > mysql-two-info.sql 


2. 直接 备份 数据 库 文件 

另 一 种 不 涉及 mysqldump 备份 数据 库 和 表 的 方式 是 直接 备份 数据 库 文件 。 通常 情况 下 ,使 用 诸 
如 Linux 中 的 cp、tar 或 cpio 实用 程序 ， 或 Windows 中 的 复制 、 粘 贴 命令 来 完成 该 操作 。 这 里 需要 
注意 的 是 ， 当 使 用 其 中 一 种 直接 备份 方法 时 ， 必 须 保证 表 不 在 被 使 用 。 如 果 服 务 器 正在 备份 时 其 中 
一 个 表 发 生 改 变 ， 那 么 这 次 备份 就 失去 了 意义 。 所 以 保证 备份 完整 性 的 最 好 方法 是 关闭 服务 器 后 ， 
再 备份 文件 ， 然 后 重启 服务 器 。 如 果 用 户 不 想 关 闭 服务 器 , 则 需要 在 执行 表 检 查 的 同时 锁定 服务 器 。 
由 于 该 种 方法 比较 简单 ， 在 此 不 再 详细 介绍 。 


10.1.11 恢复 数据 库 


数据 库 损坏 的 发 生 有 很 多 原因 ， 程 度 也 不 同 ， 有 时 可 能 仅 损坏 一 两 个 表 〈 如 掉 电 )， 而 有 时 可 
能 必须 蔡 换 整个 数据 目录 (如 磁盘 损坏 )。 如 果 表 损坏 但 没 丢 失 ， 用 户 可 以 尝试 用 myisamchk 或 
isamchk 修复 它们 。 但 在 某 些 情况 下 必须 进行 恢复 ， 比 如 用 户 错误 地 删除 了 数据 库 或 表 。 

恢复 过 程 涉及 两 种 信息 源 : 备份 文件 和 更 新 日 志 。 备 份 文件 将 表 恢 复 到 实施 备份 时 的 状态 ， 然 
而 一 般 表 在 备份 与 发 生 问 题 之 间 的 时 间 内 已 经 被 修改 ， 更 新 日 志 包 含 了 用 于 进行 这 些 修改 的 查询 。 
用 户 可 以 使 用 日 志文 件 作为 mysql 的 输入 来 重复 查询 。 这 也 是 为 什么 要 启用 更 新 日 志 的 原因 。 

恢复 过 程 视 用 户 必须 恢复 的 信息 多 少 而 不 同 。 实 际 上 ， 恢 复 整 个 数据 库 比 单个 表 更 容易 ， 因 为 
对 于 数据 库 运 用 更 新 日 志 比 单个 表 容 易 。 

1. 恢复 整个 数据 库 

首先 ， 如 果 用 户 想 恢复 的 数据 库 是 包含 授权 表 的 MySQL 数据 库 ， 那 么 需要 用 --skip-grant-table 
选项 运行 服务 器 。 和 否则 ， 将 不 能 找到 授权 表 。 在 已 经 恢复 表 后 ， 要 执行 mysqladmin flush-privileges 
告诉 服务 器 装载 授权 标识 并 使 用 它们 。 

用 最 新 的 备份 文件 重 装 数据 库 。 如 果 用 户 使 用 mysqldump 备份 的 数据 库 文 件 ， 那 么 将 它 作为 
MySQL 的 输入 。 如 果 用 户 用 直接 从 数据 库 复 制备 份 的 文件 ， 那 么 将 它们 直接 复制 回 数据 库 目录 ， 
这 里 需要 注意 的 是 ， 此 时 用 户 需要 在 恢复 文件 之 前 关闭 MySQL 服务 器 ， 然 后 重启 它 。 

使 用 更 新 日 志 重 复 操 作 备份 以 后 的 修改 数据 库 来 执行 表 的 查询 。 对 于 任何 可 使 用 的 更 新 日 志 ， 
将 它们 作为 MySQL 的 输入 。 指 定 --one-database 选项 使 得 MySQL 只 执行 用 户 希 望 恢复 的 数据 库 的 
查询 。 如 果 用 户 需 要 运用 所 有 的 更 新 日 志文 件 , 那么 可 以 在 包含 日 志 的 目录 下 使 用 如 下 所 示 的 命令 : 


%>1s -t -r -1 update.[0-9]* | xargs cat | mysql --one-database db name; 
ls 命令 生成 更 新 日 志文 件 的 一 个 单列 列表 ， 根 据 服务 器 产生 它们 的 次 序 排序 ， 这 里 需要 注意 ， 


如 果 用 户 修改 任何 一 个 文件 ， 将 改变 排序 次 序 ， 这 导致 更 新 日 志 中 错误 的 次 序 被 运用 。 如 果 用 户 只 
想 使 用 某 几 个 更 新 日 志 , 可 以 进行 如 下 操作 。 例如 , 自从 备份 以 来 产生 的 更 新 日 志 被 命名 为 update.1、 


update.2、update.3 等 ， 用 户 可 以 这 样 重新 运行 : 


%>mysql --one-database db name < update.1; 
$>mysql --one-database db name < update.2; 
%>mysql --one-database db name < update.3; 


如 果 用 户 正 在 实施 恢复 且 使 用 更 新 日 志 恢 复 某 一 个 错误 ， 是 由 DROP DATABASE、DROP 
TABLE 或 DELETE 语句 造成 丢失 的 信息 ， 在 运用 更 新 日 志 之 前 ， 要 保证 从 其 中 删除 这 些 语句 。 

2. 恢复 单个 表 

恢复 单个 表 较 为 复杂 。 如 果 用 户 使 用 mysqldump 备份 的 数据 库 文件 ， 并 且 它 不 包含 用 户 所 需要 
的 表 数 据 ， 那 么 需要 从 相关 行 中 提取 它们 并 将 它们 用 作 MySQL 的 输入 。 这 是 容易 的 部 分 。 难 的 部 
分 是 只 需要 使 用 该 表 更 新 日 志 中 的 一 部 分 ,这 时 可 以 使 用 mysql find rows 实用 程序 来 从 更 新 日 志 中 
提取 多 行 查询 。 

另 一 个 方法 是 使 用 另 一 台 服 务 器 恢复 整个 数据 库 ， 然 后 复制 想 要 的 表 文 件 到 原 数据 库 中 。 当 用 
户 将 文件 复制 到 数据 库 目录 时 ， 要 确保 原 数据 库 的 服务 器 关闭 。 


10.2 ”使 用 MySQL 数据 库 


使 用 MySQL 数据 库 最 主要 的 是 对 数据 库 中 表 的 操作 ， 包 括 向 表 中 插入 数据 、 对 表 中 数据 进行 
查询 、 编 辑 已 经 存在 的 记录 及 删除 某 些 不 再 需要 的 记录 。 这 是 使 用 MySQL 数据 库 的 最 基本 操作 ， 
也 是 进行 PHP 数据 库 编程 的 基础 。 


10.2.1 插入 数据 


INSERT 语句 用 于 向 一 个 已 有 的 表 中 插入 新 的 数据 (记录 )。 其 中 ，INSERT...VALUES 和 
INSERT..SET 形式 的 语句 根据 明确 指定 的 值 插入 数据 。INSERT..SELECT 形式 的 语句 插入 从 其 他 表 
中 选 出 的 数据 。 下 面 对 它 们 进行 分 别 介绍 。 

1， 根 据 指 定 值 插入 数据 

根据 指定 值 插入 数据 主要 有 两 种 形式 的 语句 :INSERT...VALUES 和 INSERT...SET， 这 里 主要 
介绍 INSERT...VALUES 语句 的 使 用 ， 它 的 使 用 格式 如 下 所 示 : 


INSERT [LOW PRIORITY | DELAYED | HIGH PRIORITY] [IGNORE] 
[INTO] tbl name [(col name,...)] 
VALUES. (texzbr | DEFAOIP s)he (2s) 
[ ON DUPLICATE KEY UPDATE col name=expr, ... ] 


其 中 第 一 行 和 第 三 行 是 必需 的 ， 其 他 可 选 。 第 一 行 和 第 二 行 构成 了 INSERT INTO 子 句 的 一 部 
分 。 在 这 个 子 句 中 ， 必 须 标识 准备 向 其 插入 数据 的 表 的 名 称 。 表 名 跟 在 INSERT INTO 关键 字 的 后 
面 。 然 后 可 以 标识 出 将 要 接收 数据 的 表 中 的 列 名 ， 可 以 指定 一 列 或 多 列 ， 所 有 这 些 列 都 必须 放 在 圆 
括号 内 。 如 果 指 定 的 列 不 止 一 个 ， 则 必须 用 逗号 把 这 些 列 隔 开 。 如 果 指定 了 列 名 ， 那 么 在 目标 表 中 
所 有 未 被 指定 的 列 必须 支持 空 值 ， 或 者 必须 指定 默认 值 ; 否则 ， 插 入 记录 将 会 出 错 。 
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语法 的 第 三 行 是 VALUES 子 句 ， 必 须 指定 一 个 或 多 个 要 插入 表 中 的 值 。 这 些 值 必 须 放 在 圆 括号 
内 ; 如果 指 定 的 值 不 止 一 个 ， 那 么 它们 之 间 必 须 用 逗号 隔 开 。 另 外 ， 使 用 该 语句 还 必须 满足 下 列 
要 求 : 

(1) 如 果 在 INSERT INTO 子 句 中 没有 指定 列 名 ， 那 么 表 中 的 每 一 列 必须 有 一 个 值 ， 其 次 序 必 
须 与 在 表 中 定义 的 次 序 相同 。 

(2) 如 果 在 INSERT INTO 子 句 中 指定 了 列 名 ， 那 么 每 一 个 指定 的 列 只 能 有 一 个 值 ， 且 值 的 次 
序 必须 与 在 INSERT INTO 子 句 中 定义 的 次 序 相同 ， 但 是 列 名 和 值 的 次 序 不 必 与 表 定 义 中 列 的 次 序 
相同 。 

(3) 每 个 带 有 字符 串 数据 类 型 的 值 必须 放 在 单 引号 中 。 

如 果 使 用 LOW_PRIORITY 关键 词 ， 则 INSERT 的 执行 被 延迟 ， 直 到 没有 其 他 客户 端 从 表 中 
读 取 为 止 。 当 原 有 客户 端正 在 读 取 时 ， 有 些 客户 端 刚 开 始 读 取 ， 这 些 客户 端 也 被 包括 在 内 。 此 时 ， 
INSERT LOW_PRIORITY 语句 等 候 。 因此 , 在 读 取 量 很 大 的 情况 下 , 发 出 INSERTLOW _PRIORITY 
语句 的 客户 端 有 可 能 需要 等 待 很 长 一 段 时 间 。 

如 果 使 用 DELAYED 关键 词 ， 则 服务 器 会 把 待 插入 的 行 放 到 一 个 缓冲 器 中 ， 而 发 送 INSERT 
DELAYED 语句 的 客户 端 会 继续 运行 。 如 果 表 正在 被 使 用 ， 则 服务 器 会 保留 这 些 行 。 当 表 空 闲 时 ， 
服务 器 开始 插入 行 ， 并 定期 检查 是 否 有 新 的 读 取 请 求 。 如 果 有 新 的 读 取 请 求 ， 则 被 延迟 的 行 被 延缓 
执行 ， 直 到 表 再 次 空闲 时 为 止 。 

如 果 指 定 了 HIGH PRIORITY, 同时 服务 器 --low-priority-updates 选项 启动 , 则 HIGH PRIORITY 
将 覆盖 --low-priority-updates 选项 。 这 么 做 会 导致 同时 进行 的 插入 操作 被 取消 。 

如 果 在 INSERT 语句 中 使 用 IGNORE 关键 词 ， 则 在 执行 语句 时 出 现 的 错误 被 当 作 警 告 处 理 。 例 
如 ， 没 有 使 用 IGNORE 时 ， 如 果 一 个 行 复制 了 原 有 的 UNIQUE 索引 或 PRIMARY KEY 值 ， 会 导致 
出 现 重复 关键 字 错 误 ， 语 句 执行 失败 。 使 用 IGNORE 时 ， 该 行 仍然 未 被 插入 ， 但 是 不 会 出 现 错误 。 
IGNORE 未 被 指定 时 ， 如 果 数 据 转化 引发 错误 ， 则 会 使 语句 执行 失败 。 使 用 IGNORE 后 ， 无 效 数据 
被 调整 到 最 接近 的 值 , 并 被 插入 ; 此 时 ,生成 警告 ,但 是 语句 执行 不 会 失败 .用户 可 以 使 用 mysql_info0 
函数 测定 有 多 少 行 被 插入 表 中 。 

如 果 指 定 了 ON DUPLICATE KEY UPDATE， 并 且 插 入 行 后 会 导致 在 一 个 UNIQUE 索引 或 
PRIMARY KEY 中 出 现 重复 值 ， 则 执行 旧 行 UPDATE。 例 如 ， 如 果 列 numl 被 定义 为 UNIQUE， 并 
且 包 含 值 1， 则 以 下 两 个 语句 具有 相同 的 效果 : 

mysql> INSERT INTO MyNum (numl,num2,num3) VALUES (1,2,3) 

-> ON DUPLICATE KEY UPDATE num3=num3+17 


和 
mysql> UPDATE MyNum SET num3=num3+1 WHERE numl=1; 


如 果 数 据 作为 新 记录 被 插入 ， 则 受 影响 行 的 值 为 1 如 果 原 有 的 记录 被 更 新 ， 则 受 影响 行 的 值 
为 2。 

2. 使 用 SELECT 语句 插入 数据 

通过 SELECT 语句 生成 结果 集 ， 然 后 结合 INSERT 语句 把 结果 集 插 入 指定 的 表 中 ， 这 种 方法 用 
于 插入 的 数据 不 确定 ， 并 且 具 有 一 些 特性 时 。INSERT...SELECT 语句 的 使 用 格式 如 下 所 示 : 


INSERT [LOW PRIORITY | HIGH PRIORITY] [IGNORE] 


[INTO] tbl name [(col name,...)] 
SELECT --- 
[ ON DUPLICATE KEY UPDATE col_name=expr，--- ] 


组 合 的 INSERT...SELECT 语句 可 以 把 其 他 数据 表 的 行 添加 到 现 有 的 表 中 。 使 用 INSERT... 
SELECT 语句 比 使 用 多 个 单行 的 INSERT 语句 效率 要 高 得 多 。 使 用 INSERT...SELECT 语句 应 满足 以 
下 要 求 : 

(1) 明确 指定 IGNORE， 用 于 忽略 会 导致 重复 关键 字 错 误 的 记录 。 

(2) 不 要 同时 使 用 DELAYED 和 INSERT...SELECT。 

(3) INSERT 语句 的 目标 表 会 显示 在 查询 的 SELECT 部 分 的 FROM 子 句 中 (在 有 些 旧 版 本 的 
MySQL 中 不 会 出 现 这 种 情况 )。 

(4) 为 了 确保 二 进 制 日 志 可 以 被 用 于 再 次 创建 原 表 ，MySQL 不 允许 在 INSERT...SELECT 运行 
期 间 同 时 进行 插入 操作 。 

(5) 不 能 在 向 一 个 表 插 入 数据 的 同时 ， 又 在 一 个 子 查询 中 从 同一 个 表 中 选择 。 

在 ON DUPLICATE KEYUPDATE 的 值 部 分 中 ,只 要 不 使 用 SELECT 部 分 中 的 GROUP BY, 用 
户 就 可 以 引用 在 其 他 表 中 的 列 。 这 里 需要 注意 的 是 ， 用 户 必 须 使 值 部 分 中 的 非 唯一 列 的 名 称 符合 要 
求 。 

另外 ， 用 户 可 以 使 用 REPLACE 替代 INSERT， 来 覆盖 旧 行 。 对 于 包含 唯一 关键 字 值 ， 并 复制 
了 旧 行 的 新 行 ， 在 进行 处 理 时 ，REPLACE 可 以 作为 INSERT IGNORE 的 同类 子 句 : 新 行 被 用 于 蔡 
换 旧 行 ， 而 不 是 被 丢弃 。 


10.2.2 ”查询 数据 


查询 数据 是 指 根据 用 户 的 需要 以 一 种 可 读 的 方式 从 数据 库 中 提取 所 需 数据 。 在 MySQL 数据 库 
中 ， 数 据 查询 功能 是 通过 SELECT 语句 来 实现 的 。SELECT 语句 可 以 从 数据 库 中 按照 用 户 的 要 求 查 
询 数据 ， 并 将 查询 结果 以 表格 的 形式 输出 数据 。 

SELECT 语句 是 一 个 查询 表达 式 ， 它 包括 SELECT、FROM、WHERE、GROUP BY 和 ORDER 
BY 子 句 。SELECT 语句 具有 数据 查询 、 统 计 、 分 组 和 排序 的 功能 ， 它 可 以 精确 地 对 数据 库 进行 查 
询 ， 也 可 以 进行 模糊 查询 。SELECT 语句 的 完整 使 用 格式 相当 复杂 ， 这 里 只 列 出 常见 的 部 分 ， 如 下 
所 示 : 

SELECT [ALL | DISTINCT | DISTINCTROW ] select list 

[ INTO new table ] 

FROM table source 

[ WHERE search conditions ] 

[ GROUP BY group by expression] 

[ HAVING search conditions] 

[ ORDER BY order expression [ RSC | DESC ] ] 


查询 语句 的 功能 是 从 FROM 子 句 指定 的 数据 源 (基本 表 或 视图 组 ) 中 ,选择 满足 元 组 选择 条 件 


的 元 组 数据 ， 并 对 它们 进行 分 组 、 统 计 、 排 序 和 投影 ， 形 成 查询 结果 集 。 在 查询 语句 中 共有 5 种 子 
句 ， 其 中 SELECT 和 FROM 语句 为 必 选 子 句 ， 而 WHERE、GROUP BY 和 ORDER BY 子 句 为 任 选 
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子 句 。 下 面 对 使 用 格式 中 的 各 部 分 进行 一 下 简要 介绍 ， 具 体 如 表 10-6 所 示 。 
表 10-6 ”SELECT 语句 格式 各 部 分 说 明 


语句 部 分 说 明 

SELECT 子 句 用 来 指定 由 查询 返回 的 列 ， 并 且 各 列 在 SELECT 子 句 中 的 顺序 决定 了 它们 在 
结果 集中 输出 的 位 置 

ALL | DISTINCT | 用 来 标识 在 查询 结果 集中 对 相同 行 处 理 的 方式 。 关键 字 ALL 表示 返回 查询 结 

DISTINCTROW 果 集 的 所 有 行 ， 其 中 包括 重复 行 ; 关键 字 DISTINCT 和 DISTINCTROW 表示 
若 结果 集中 有 相同 行 则 只 保留 显示 一 行 。 默 认 值 为 ALL 

Select_list 目标 列 。 若 有 多 个 目标 列 ， 各 列 名 之 间 用 逗号 隔 开 。 若 需要 返回 所 有 列 的 数 
据 信息 ， 则 可 用 * 表 示 

INTO new_table 表示 创建 一 个 新 表 ， 表 名 为 new_table， 表 的 内 容 为 查询 结果 集 

FROM table_source 用 来 指定 数据 源 (table_ source)， 如 基本 表 或 视图 

WHERE search conditions 用 来 指定 限定 返回 行 的 搜索 条 件 〈search conditions) 

GROUP BY group by_expression | 用 来 指定 查询 结果 的 分 组 条 件 

HAVING search_conditions 用 来 指定 组 或 聚合 的 搜索 条 件 

ORDER BY order_expression 用 来 指定 结果 集 的 排序 方式 。ASC 表示 结果 集 以 升序 排列 ，DESC 表示 结果 

[ASCIDESC] 集 以 降序 排列 


在 使 用 SELECT 语句 时 ， 需 要 遵守 下 列 两 条 规则 : 
(1) 虽然 SQL 是 一 种 自由 格式 语言 ， 但 也 必须 按 句 法 顺序 来 处 理 SELECT 语句 中 的 子 句 。 例 


如 ，GROUP BY 子 句 必须 在 ORDER BY 子 句 前 出 现 ， 否 则 会 出 现 语法 
(2) 当 给 SELECT 语句 指定 目标 列 时 ， 如 果 使 用 的 列 名 不 明确 ， 就 需要 通过 给 该 列 名 命名 的 方 
式 来 确定 该 列 的 数据 源 。 例 如 ， 在 同一 数据 库 中 的 多 个 基本 表 (如 message、reply) 中 都 存在 着 名 
为 msgid 的 列 ， 当 进行 多 表 操 作 时 ， 如 果 单 使 用 msgid 指定 选择 条 件 ， 就 会 出 现 歧义 ， 所 以 需要 通 
过 表 名 来 明确 所 要 指定 的 msgid 列 ， 即 用 message.name 和 reply.name 表示 。 
现 假设 要 从 表 message 中 查询 出 字段 msgid 值 为 2 的 所 有 记录 ， 然 后 显示 这 些 记录 的 repid、 
recontent 及 retime 字段 的 值 ， 并 按 字段 repid 降序 排列 。 具 体 语句 如 下 所 示 : 


mysql> SELECT repid, recontent, retime 
-> FROM reply 
-> WHERE msgid=1 
-> ORDER BY repid DESC; 


成 功 执行 上 述 语句 ， 结 果 如 图 10-22 国史 区 
所 示 。 


10.2.3 ”编辑 记录 


数据 被 添加 到 数据 表 后 ， 随 着 时 间 的 推 
移 某 些 数据 会 发 生变 化 , 所 以 经 常 需要 修改 ， 
如 用 户 的 联系 方式 、 商 品 的 价格 等 。 在 
MySQL 中 ， 对 数据 的 修改 也 称 为 编辑 记录 ， 是 通过 使 用 UPDATE 语句 来 实现 的 。 其 使 用 格式 如 下 
所 示 : 


图 10-22 执行 结果 


UPDATE [LOW PRIORITY] [IGNORE] table references 
SET col namel=exprl [, col name2=expr2 ...] 
[WHERE where definition] 
可 以 看 出 ，UPDATE 子 句 和 SET 子 句 是 必需 的 ， 而 WHERE 子 句 是 可 选 的 。 在 UPDATE 子 句 
中 ， 必 须 指定 将 要 更 新 的 数据 表 的 名 称 。 在 SET 子 句 中 ， 必 须 指定 一 个 或 多 个 子 句 表达 式 。 
使 用 UPDATE 语句 可 以 更 改 表 中 单行 、 多 行 或 者 表 或 视图 中 所 有 行 的 数值 。 用 户 既 可 以 根据 自 
己 表 的 数据 进行 更 新 ， 也 可 以 根据 其 他 表 的 数据 进行 更 新 。 
如 果 用 户 把 已 定义 为 NOT NULL 的 列 更 新 为 NULL， 则 该 列 被 设置 到 与 列 类 型 对 应 的 默认 值 ， 
并 且 累 加 警告 数 。 对 于 数字 类 型 ， 默 认 值 为 0; 对 于 字符 串 类 型 ， 默 认 值 为 空 字符 串 ("); 对 于 日 期 
和 时 间 类 型 ， 默 认 值 为 “zero” 值 。 
当 使 用 UPDATE 语句 时 ， 应 该 注意 以 下 的 事项 和 原则 : 
。 用 WHERE 子 句 指定 需要 更 新 的 行 ， 用 SET 子 句 指定 新 值 。 
。 UPDATE 无 法 更 新 标识 列 。 
。 如 果 行 的 更 新 违反 了 约束 或 规则 ， 比 如 违反 了 列 的 NULL 设置 , 或 者 新 值 是 不 兼容 的 数据 类 
型 ， 则 将 取消 该 语句 ， 并 返回 错误 ， 不 会 更 新 任何 记录 。 
。 每 次 只 能 修改 一 个 表 中 的 数据 。 
。 可 以 同时 把 一 列 或 多 列 、 一 个 变量 或 多 个 变量 放 在 一 个 表达 式 中 。 例 如 ， 某 个 表达 式 可 以 是 
计算 得 出 的 ( 如 number*2)， 也 可 以 是 两 列 相 加 。 
现 假设 要 从 表 message 中 查询 出 字段 repid 值 为 3 的 所 有 记录 ， 然 后 修改 该 记录 的 recontent 字 
段 的 值 为 : Hello IT 在 中 国 。 具 体 语句 如 下 所 示 : 
mysql> UPDATE reply 


-> SET recontent = 'Hello IT 在 中 国 ' 
-> WHERE repid=3; 


成 功 执行 上 述 语句 ， 将 输出 如 下 信息 : 


Query OK, 1 row affected (0.06 sec) 
Rows matched: 1 Changed: 1 Warnings: 0 


为 了 验证 是 否 修 改 成 功 ， 可 执行 SELECT 查询 ， 结 果 如 图 10-23 所 示 。 
10.2.4 删除 记录 
当 数 据 表 中 的 记录 不 再 需要 时 ， 如 学 生 


从 所 在 学 校 毕业 、 员 工 离开 所 在 公司 ， 那 么 
就 可 以 从 学 生 信 息 表 、 员 工 信 息 表 中 删除 该 


学 生 或 员工 对 应 的 记录 。 在 PHP 中 ,删除 记 
录 是 通过 使 用 DELETE 语句 来 实现 的 。 其 使 图 10-23 查询 修改 结果 
用 格式 如 下 所 示 : 


DELETE [LOW PRIORITY] [QUICK] [IGNORE] FROM tb] name 
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[WHERE where definition] 
[ORDER BY ...] 
[LIMIT row count] 


上 述 使 用 格式 中 , tbl_ name 表 中 有 些 行 满足 由 where_definition 给 定 的 条 件 。 DELETE 用 于 删除 
这 些 行 ， 并 返回 被 删除 的 记录 的 数目 。 
如 果 编 写 的 DELETE 语句 中 没有 WHERE 子 句 , 则 所 有 的 行 都 被 删除 。 如 果 用 户 不 想 知道 被 删 
除 的 行 的 数目 时 ， 可 以 使 用 TRUNCATE TABLE。 
如 果 删 除 的 行 中 包括 用 于 AUTO_INCREMENT 列 的 最 大 值 ， 则 该 值 被 重新 用 于 BDB 表 , 但 是 
不 会 被 用 于 MyISAM 表 或 InnoDB 表 。 如 果 用 户 在 AUTOCOMMIT 模式 下 使 用 DELETE FROM 
tbl name (不 含 WHERE 子 句 ) 删除 表 中 的 所 有 行 ， 则 对 于 所 有 的 表 类 型 ( 除 InnoDB 和 MyISAM 
外 )， 序 列 将 重新 编排 。 
下 面 对 上 述 格式 中 的 个 别 参数 说 明 如 下 : 
。 如 果 指 定 LOW _PRIORITY， 则 DELETE 的 执行 被 延迟 ， 直 到 没有 其 他 客户 端 读 取 该 表 时 再 
执行 。 
。 对 于 MyISAM 表 ， 如 果 使 用 QUICK 关键 词 ， 则 在 删除 过 程 中 ， 存 储 引 擎 不 会 合并 索引 端 结 
点 ， 这 样 可 以 加 快 部 分 种 类 的 删除 操作 的 速度 。 
。 在 删除 行 的 过 程 中 ，IGNORE 关键 词 会 使 MySQL 忽略 所 有 的 错误 ( 在 分 析 阶段 遇 到 的 错误 
会 以 常规 方式 处 理 )。 由 于 使 用 本 选项 而 被 忽略 的 错误 会 作为 警告 返回 。 
如 果 DELETE 语句 包括 一 个 ORDER BY 子 句 ， 则 各 行 按照 子 句 中 指定 的 顺序 进行 删除 。 此 子 
句 只 在 与 LIMIT 联 用 时 才 起 作用 。 例 如 ， 以 下 子 句 用 于 查找 与 WHERE 子 句 对 应 的 行 ， 使 用 time 
进行 分 类 ， 并 删除 第 一 (最 旧 的 ) 行 。 具 体 如 下 所 示 : 


DELETE FROM message 

WHERE username = "SongYan'" 
ORDER BY time 

LIMIT 1; 


除了 前 面 的 删除 方式 外 , 也 可 以 在 一 个 DELETE 语句 中 指定 多 个 表 , 根据 多 个 表 中 的 特定 条 件 ， 
从 一 个 表 或 多 个 表 中 删除 行 。 但 是 不 能 在 一 个 多 表 DELETE 语句 中 使 用 ORDER BY 或 LIMIT。 具 
体 示例 如 下 所 示 : 

DELETE tablel, table2 FROM tablel, table2, table3 WHERE tablel.id=table2.id AND 

table2.id=table3.id; 

或 


DELETE FROM tablel, table2 USING tablel, table2, table3 WHERE tablel.id=table2.id 
AND table2.id=table3.id; 


对 于 第 一 个 DELETE 语句 ， 只 删除 列 于 FROM 子 句 之 前 的 表 中 的 对 应 的 行 。 对 于 第 二 个 
DELETE 语句 ， 只 删除 列 于 FROM 子 句 之 中 (在 USING 子 句 之 前 ) 的 表 中 的 对 应 的 行 。 利 用 该 方 
法 ， 用 户 可 以 同时 删除 多 个 表 中 的 行 ， 并 使 用 其 他 的 表 进 行 搜索 。 

如 果 使 用 的 多 表 DELETE 语句 包括 InnoDB 表 ， 并 且 这 些 表 受 外 键 的 限制 ， 则 MySQL 优化 程 
序 会 对 表 进行 处 理 ， 改 变 原 来 的 从 属 关系 。 在 这 种 情况 下 ， 该 语句 出 现 错误 并 返回 到 前 面 的 步 又 。 
要 避免 此 错误 ， 应 该 从 单一 表 中 删除 ， 并 依靠 InnoDB 提供 的 ON DELETE 功能 ， 对 其 他 表 进 行 相 


应 的 修改 。 


10.3 MySQL 的 高 级 应 用 


本 章 介绍 MySQL 的 高 级 应 用 ， 主 要 包括 事务 与 存储 过 程 ， 其 中 事务 〈transaction) 是 作为 一 个 
单元 的 一 组 有 序 的 数据 库 操作 。 一 个 事务 由 一 个 或 多 个 完成 一 组 相关 行为 的 SQL 语句 组 成 。 而 存储 
过 程 (stored procedure) 是 一 组 为 了 完成 特定 功能 的 SQL 语句 集 ， 经 编译 后 存储 在 数据 库 中 。 存 储 
过 程 可 包含 程序 流 、 逻 辑 以 及 对 数据 库 的 查询 。 它 们 可 以 接收 参数 、 输 出 参数 、 返 回 单个 或 多 个 结 
果 集 以 及 返回 值 。 


10.3.1 事务 


于 事务 是 一 组 有 序 的 数据 库 操作 组 成 的 ， 所 以 如 果 组 中 的 所 有 操作 都 成 功 ， 则 认为 事务 执行 
成 功 ， 否 则 ， 即 使 只 有 一 个 操作 失败 ， 那 么 事务 便 执行 失败 。 如 果 所 有 操作 执行 成 功 ， 事 务 则 提交 
(commit)， 所 有 的 修改 都 将 生效 。 如 果 这 些 操作 中 有 一 个 执行 失败 ， 则 事务 将 回 滚 (rollback)， 数 
据 库 也 返回 到 事务 开始 前 的 状态 ， 所 有 的 修改 都 会 被 取消 。 

事务 的 主要 作用 是 保证 数据 库 的 完整 性 。 因此 , 从 保证 数据 库 完整 性 出 发 事务 应 具有 4 个 特性 : 
原子 性 〈Atomicity)、 一 致 性 (Consistency)、 隔 离 性 (Isolation) 及 持久 性 (Durability)， 简 称 ACID 。 
下 面 分 别 加 以 介绍 ， 如 表 10-7 所 示 。 


表 10-7 事务 的 4 个 特性 
特 性 说 了 明 
原子 性 | 原子 性 表示 一 个 事务 要 么 被 提交 ， 要 么 被 完整 地 回 滚 。 对 用 户 而 言 ， 所 有 的 中 间 阶 段 都 是 透明 的 。 一 
个 事务 中 的 所 有 修改 要 么 全 部 实现 ， 要 么 全 部 被 回 滚 

一 致 性 | 数据 库 在 事务 开始 和 结束 时 必须 一 致 ， 事 务 中 任 选任 务 数据 的 变化 都 必须 符合 数据 定义 规则 。 另 外 ， 
事务 结束 时 数据 库 内 的 所 有 结构 都 必须 正确 

隔离 性 | 在 事务 处 理 过程 中 暂时 不 一 致 的 数据 不 能 被 其 他 事务 应 用 ， 直 到 数据 再 次 一 致 。 也 就 是 说 ， 当 某 项 事 
务 使 数据 不 一 致 时 ， 其 他 事务 将 不 能 访问 该 事务 中 不 一 致 的 数据 


持久 性 | 一 旦 由 事务 引发 的 变化 发 生 ， 这些 变 化 就 被 保护 起 来 ， 即 使 硬件 和 应 用 程序 发 生 错误 ， 这些 数据 也 会 
可 靠 一 致 


1. MySQL 事务 概述 

MySQL 自 版 本 3.23.34a 开始 支持 事务 ， 在 MySQL 4.0 及 以 上 版 本 中 均 默认 地 启用 事务 。 在 
MySQL 中 有 两 个 MySQL 表 处 理 器 (InnoDB 和 BDB) 支持 事务 。 其 中 ，InnoDB 是 一 个 健壮 的 事 
务 型 存储 引擎 ， 该 方案 已 被 证 明 是 事务 型 应 用 程序 的 流行 且 有 效 的 解决 方案 。 自 版 本 4.1 开始 ， 
InnoDB 就 被 MySQL Windows 安装 包 指 定 为 默认 引擎 。 

而 BDB (Berkeley DB) 存储 引擎 由 Sleepycat 软件 公司 (http://www.sleepycat.com) 创建 和 维护 ， 
它 第 一 次 将 事务 功能 引入 MySQL。 随 着 集成 更 紧密 的 InnoDB 引擎 的 引入 ,该 引擎 的 优势 已 经 显著 
减少 ,致使 BDB 表 驱 动 程序 已 经 从 MySQL 的 二 进 制 分 发 包 中 删除 .但 是 ,用 户 可 以 通过 安装 MySQL 
的 数据 库 服务 器 的 最 大 版 本 , 继续 使 用 BDB 存储 引擎 。 此 外 , 用户 可 以 通过 从 源 代码 包 配 置 MySQL 
时 ， 通 过 设置 --with-berkeley-db 选项 启用 这 个 表 类 型 。 
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因为 InnoDB 存储 引擎 在 MySQL 4.0 及 以 上 版 本 中 是 默认 启动 的 ,用 户 可 以 通过 执行 如 下 命令 ， 
来 验证 InnoDB 表 是 否 可 用 : 

mysql> SHOW VARIABLES LIKE ~ Ny5QL Comnand Line Client 

'have_ innoDB'; 

具体 执行 结果 如 图 10-24 所 示 。 

如 果 用 户 没 有 看 到 类 似 于 图 10-24 所 示 
四 的 结果 ， 那 么 用 户 需要 将 MySQL 分 发 包 升 
级 到 提供 InnoDB 支持 的 版 本 。 如 果 用 户 使 
用 的 是 版 本 3.23.34 与 版 本 4.0 之 间 的 版 本 ， 
就 可 以 通过 --with-innodb 选项 重 编 译 
MySQL 来 启用 支持 。 另 外 ， 如 果 用 户 使 用 
的 是 MySQL 4.1.0 或 更 高 的 版 本 ， 则 可 以 使 
用 新 命令 SHOW TABLE TYPES 来 确定 是 否 
支持 某 个 表 类 型 。 执 行 上 述 语句 的 部 分 表 类 
型 如 图 10-25 所 示 。 

2. InnoDB 启动 选项 
下 面 介 绍 与 InnoDB 相关 的 服务 器 选 图 10-25 MySQL 支持 的 表 类 型 
项 ， 所 有 这 些 选项 可 以 以 --opt name=value 
的 形式 在 命令 行 或 在 选项 文件 中 被 指定 ， 有 具体 如 表 10-8 所 示 。 
表 10-8 InnoDB 相 关 的 服务 器 选项 
选 项 说 了 明 
innodb_additional_ mem pool size | InnoDB 用 来 存储 数据 目录 信息 及 其 他 内 部 数据 结构 的 内 存 池 的 大 小 。 用 户 应 


用 程序 中 的 表 越 多 ， 需 要 在 这 里 分 配 的 内 存 越 多。 如 果 InnoDB 用 完了 这 个 池 
内 的 内 存 , 就 开始 从 操作 系统 分 配 内 存 , 并 且 往 MySQL 错误 日 志 写 警告 信息 。 
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默认 值 是 1MB 

innodb_autoextend_increment 当 自 动 扩 展 表 空间 被 填 满 时 ， 为 扩展 而 增加 的 空间 (以 MB 为 单位 )。 默 认 值 
是 8。 这 个 选项 可 以 在 运行 时 作为 全 局 系统 变量 而 改变 

innodb_buffer_pool size InnoDB 用 来 缓存 它 的 数据 和 索引 的 内 存 缓冲 区 的 大 小 。 把 这 个 值 设 得 越 高 ， 


访问 表 中 数据 需要 的 磁盘 IO 越 少 。 在 一 个 专用 的 数据 库 服务 器 上 ， 可 以 设置 
这 个 参数 达 物 理 内 存 大 小 的 80%。 尽 管 如 此 ， 还 是 不 要 把 它 设置 得 太 大 ， 因 为 
对 物理 内 存 的 竞争 可 能 在 操作 系统 上 导致 内 存 调度 

innodb_checksums InnoDB 在 所 有 对 磁盘 的 页 面 读 取 上 使 用 校 验 和 验证 ， 以 确保 额外 容错 防止 硬 
件 损坏 或 数据 文件 丢失 。 尽 管 如 此 ， 在 一 些 少见 的 情况 下 (比如 运行 标准 检查 
时 ) 这 个 额外 的 安全 特征 是 不 必要 的 。 在 这 些 情况 下 ， 这 个 选项 (默认 是 允许 
的 ) 可 以 用 --skip-innodb-checksums 来 关闭 
innodb_data_file_path 该 选项 表示 到 单独 数据 文件 和 它们 尺寸 的 路 径 。 通 过 把 innodb_data_home_dir 
连接 到 这 里 指定 的 每 个 路 径 ， 到 每 个 数据 文件 的 完整 目录 路 径 可 被 获得 。 文 件 
大 小 通过 给 尺寸 值 尾 加 M 或 G 以 MB 或 者 GB (1024MB ) 为 单位 被 指定 。 文 
件 尺寸 的 和 至 少 是 10MB。 在 一 些 操作 系统 上 ， 文 件 必须 小 于 2GB。 如 果 没 有 
指定 innodb data file path， 开 始 的 默认 行为 是 创建 一 个 单独 的 大 小 10MB 名 
为 ibdatal 的 自 扩展 数据 文件 。 在 那些 支持 大 文件 的 操作 系统 上 ， 可 以 设置 文 
件 大 小 超过 4GB， 也 可 以 使 用 原始 磁盘 分 区 作为 数据 文件 
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innodb data home dir 


innodb doublewrite 


该 选项 表示 目录 路 径 对 所 有 InnoDB 数据 文件 的 共同 部 分 。 如 果 不 设 置 这 个 值 ， 默 
认 是 MySQL 数据 目录 。 也 可 以 指定 这 个 值 为 一 个 空 字符 串 ， 在 这 种 情况 下 ， 可 以 
在 innodb data file path 中 使 用 绝对 文件 路 径 

默认 地 ，InnoDB 存储 所 有 数据 两 次 ， 第 一 次 存储 到 doublewrite 缓冲 ， 然 后 存储 到 
确实 的 数据 文件 。 这 个 选项 可 以 被 用 来 禁止 这 个 功能 。 类 似 于 innodb_checksums， 
这 个 选项 默认 是 允许 的 ; 因为 标准 检查 或 在 对 顶级 性 能 的 需要 超过 对 数据 完整 性 或 
可 能 故障 的 关注 之 时 ， 这 个 选项 可 用 --skip-innodb-doublewrite 来 关闭 


innodb file io _ threads 


InnoDB 中 文件 IO 线程 的 数量 。 通 常情 况 下 ， 这 个 参数 是 默认 的 ， 默 认 值 是 4, 但 
是 大 数值 对 Windows 磁盘 IO 有 益 。 在 UNIX 上 ， 增 加 这 个 数 没有 效果 ，InnoDB 
总 是 使 用 默认 值 


innodb file per table 


innodb flush log at trx_ 
commit 


innodb lock wait timeout 


innodb log arch dir 
innodb log archive 


innodb log_buffer size 


innodb log file_ size 


该 选项 致使 mnoDB 用 自己 的 .ibd 文件 为 存储 数据 和 索引 创建 每 一 个 新 表 ， 而 不 是 
在 共享 表 空间 中 创建 

该 选项 被 设置 为 0， 日 志 缓冲 每 秒 一 次 地 被 写 到 日 志文 件 ， 并 且 对 日 志文 件 做 到 磁 
盘 操 作 的 刷新 , 但 是 在 一 个 事务 提交 时 不 做 任何 操作 。 当 这 个 值 为 1 (默认 值 ) 时 ， 
在 每 个 事务 提交 时 ， 日 志 缓冲 被 写 到 日 志文 件 ， 对 日 志文 件 做 到 磁盘 操作 的 刷新 。 
当 设置 为 2 时 ， 在 每 个 事务 提交 时 ， 日 志 缓 冲 被 写 到 文件 ， 但 不 对 日 志文 件 做 到 磁 
盘 操 作 的 刷新 

InnoDB 事务 在 被 回 滚 之 前 可 以 等 待 一 个 锁定 的 超时 秒 数 。 InnoDB 在 它 自己 的 锁定 
表 中 自动 检测 事务 死 锁 并 且 回 滚 事务 。InnoDB 用 LOCK TABLES 语句 注意 到 锁定 
设置 。 默认 值 是 50 秒 。 例 如 ， 为 在 一 个 复制 建立 中 最 大 可 能 的 持久 程度 和 连贯 性 ， 
应 该 在 主 服务 器 上 的 my.cnf 文件 中 使 用 innodb flush log at trx_commit=1 和 
sync-binlog=1 

如 果 使 用 日 志 档案 , 被 完整 写 入 的 日 志文 件 所 在 的 目录 也 被 归档 。 这 个 参数 值 如 果 
被 使 用 了 ， 应 该 被 设置 得 与 innodb log_group_home dir 一 样 ， 但 它 不 是 必需 的 
该 选项 的 默认 值 是 0。 因 为 MySQL 使 用 它 自己 的 日 志文 件 从 备份 来 恢复 ， 所 以 当 
前 没有 必要 来 归档 InnoDB 日 志文 件 

InnoDB 用 来 向 磁盘 上 的 日 志文 件 写 操作 的 缓冲 区 的 大 小 。 通 常情 况 下 设置 值 从 
1~8MB。 默 认 的 是 IMB。 一 个 大 的 日 志 缓冲 允许 大 型 事务 运行 而 不 需要 在 事务 提 
交 之 前 向 磁盘 写 日 志 。 因 此 ， 如 果 有 大 型 事务 ， 使 日 志 缓冲 区 更 大 以 节约 磁盘 IO 
该 选项 设置 在 日 志 组 中 每 个 日 志文 件 的 大 小 。 在 32 位 计算 机 上 日 志文 件 的 合并 大 
小 必须 少 于 4GB。 该 选项 默认 是 SMB。 通 常情 况 下 设置 值 从 1MB 到 N 分 之 一 组 
冲 池 大 小 ， 其 中 N 是 组 中 日 志文 件 的 数目 。 值 越 大 ， 在 缓冲 池 越 少 需要 检查 点 刷 
新 行为 ， 以 节约 磁盘 IO。 但 更 大 的 日 志文 件 也 意味 着 在 崩溃 时 恢复 得 更 慢 


innodb log group home dir 


该 选项 表示 到 InnoDB 日 志文 件 的 目录 路 径 。 它 必须 有 和 innodb_log arch_dir 一 样 
的 值 。 如 果 不 指定 任何 InnoDB 日 志 参 数 ， 默 认 的 是 在 MySQL 数据 目录 中 创建 两 
个 5MB 大 小 名 为 ib logfile0 和 ib logfilel 的 文件 


innodb_open files 


在 InnoDB 中 ， 这 个 选项 仅 与 用 户 使 用 多 表 空 间 时 有 关 。 它 指定 InnoDB 一 次 可 以 
保持 打开 的 .ibd 文件 的 最 大 数目 。 最 小 值 是 10。 默 认 值 300。 注 意 ， 对 .ibd 文件 的 
文件 描述 符 是 仅 对 InnoDB 的 。 它们 独立 于 那些 由 --open-files-limit 服务 器 选项 指定 
的 描述 符 ， 且 不 影响 表 缓 存 的 操作 


innodb_table locks 


InnoDB 重视 LOCK TABLES, 直到 所 有 其 他 线程 已 经 释放 所 有 对 表 的 锁定 , MySQL 
才 从 LOCK TABLE...WRITE 返回 .默认 值 是 1, 用 于 表示 LOCK TABLES 让 InnoDB 
内 部 锁定 一 个 表 。 在 使 用 AUTOCOMMIT=1 的 应 用 中 ,InnoDB 的 内 部 表 锁 定 会 导 
致死 锁 。 用 户 可 以 在 mycnf 文件 (Windows 上 是 myini 文件 ) 中 设置 
innodb table locks=0 来 消除 这 个 问题 
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3. 创建 InnoDB 表 空 间 

假设 用 户 已 经 安装 了 MySQL, 并 且 已 经 编辑 了 选项 文件 , 使 得 它 包含 必要 的 InnoDB 配置 参数 。 
在 启动 MySQL 之 前 ， 用 户 应 该 验证 为 InnoDB 数据 文件 和 日 志文 件 指定 的 目录 是 否 存在 ， 并 且 
MySQL 有 访问 这 些 目录 的 权限 。InnoDB 不 能 创建 目录 ， 只 能 创建 文件 。 另 外 还 要 检查 是 否 有 足够 
的 空间 来 存放 数据 和 日 志文 件 。 

当 创 建 InnoDB 数据 库 时 ,最 好 从 命令 提示 符 运 行 MySQL 服务 器 mysqld, 而 不 要 从 mysqld_ safe 
包装 或 作为 Windows 的 服务 来 运行 。 当 用 户 从 命令 提示 符 运 行 时 ， 可 看 见 mysqld 打印 了 什么 以 及 
发 生 了 什么 。 在 UNIX 上 ， 只 需要 调用 mysqld。 在 Windows 上 ， 使 用 --console 选项 。 

如 果 在 创建 mnoDB 表 空 间 的 过 程 中 出 现 错 误 ， 那 么 可 能 存在 如 下 所 示 的 问题 : 

。 没有 创建 一 个 InnoDB 数据 文件 目录 或 InnoDB 日 志 目 录 。 

。 mysqld 没有 访问 这 些 目录 的 权限 以 创建 文件 。 

。 mysqld 不 能 恰当 地 读 取 my.cnf 或 my.ini 选项 文件 ， 因 此 不 能 看 到 用 户 指定 的 选项 。 

。 磁盘 已 满 ， 或 者 超出 磁盘 配额 。 

。 已 经 创建 了 一 个 子 目 录 ， 它 的 名 字 与 指定 的 数据 文件 相同 。 

。 在 innodb data home dir 或 innodb data file path 中 有 一 个 语法 错误 。 

当 InnoDB 试 着 初始 化 它 的 表 空 间或 日 志文 件 时 , 如果 出 错 , 用 户 应 该 删除 InnoDB 创建 的 所 有 
文件 ， 也 就 是 所 有 ibdata 文件 和 所 有 ib_logfiles 文件 。 如果 用 户 创建 了 一 些 InnoDB 表 ， 为 这 些 表 也 
从 MySQL 数据 库 目录 删除 相应 的 .frm 文件 〈 如 果 使 用 多 重 表 空 间 ， 也 删除 任何 .ibd 文件 )。 然 后 试 
着 再 次 创建 mnoDB 数据 库 。 注 意 ， 最 好 是 从 命令 提示 符 启 动 MySQL 服务 器 ， 以 便 查 看 启动 信息 。 

4. 创建 InnoDB 类 型 的 表 

创建 InnoDB 类 型 的 表 与 创建 任何 其 他 类 型 表 的 过 程 没有 区 别 。 在 实际 应 用 中 ， 这 个 表 类 型 在 
Windows 中 是 MySQL 5.0 的 默认 表 类 型 ， 也 就 是 说 ， 如 果 在 此 平台 上 运行 MySQL 5.0 或 更 高 版 本 ， 
则 不 需要 任何 特殊 的 动作 。 只 要 使 用 CREATE TABLE 语句 ， 并 在 表 创 建 SQL 语句 中 指定 ENGINE 
= InnoDB 或 者 TYPE = InnoDB 选项 ， 然 后 创建 所 需 的 表 即 可 。 这 里 需要 注意 ， 除 非 用 
--default-table-type=InnoDB 标志 启动 MySQL 守护 进程 ， 否 则 就 需要 在 创建 时 显 式 指定 要 将 表 创 建 
为 InnoDB 类 型 。 有 具体 示例 如 下 所 示 : 

CREATE TABLE student ( 

stuid SMALLINT UNSIGNED NOT NULL AUTO INCREMENT, 
name VARCHAR(20) NOT NULL, 


PRIMARY KEY (stuid) 
) TYP=InnoDB; 


执行 上 述 创建 语句 后 , 在 相应 的 目录 中 会 存储 一 个 *.frm 文件 (在 本 示例 中 为 student.frm 文件 )， 
其 位 置 由 MySQL 的 datadir 参数 指定 ， 在 守护 进程 启动 时 定义 。 此 文件 包含 MySQL 所 需 的 数据 字 
典 信息 。InnoDB 引擎 要 求 所 有 InnoDB 数据 和 索引 信息 存储 在 表 空 间 中 。 此 表 空 间 实 际 上 包括 很 多 
独立 的 文件 (甚至 是 原始 磁盘 分 区 )， 默 认 位 于 MySQL 的 datadir 目录 中 。 也 就 是 说 ， 只 要 在 必要 
时 将 新 文件 连接 到 表 空 间 ， 就 可 以 创建 很 大 的 表 空 间 ， 远 远 超过 很 多 操作 系统 所 允许 的 最 大 文件 大 
小 。 这 些 行为 信赖 于 用 户 如 何 定义 前 面 已 经 介绍 的 相关 的 InnoDB 启动 选项 。 

5. 事务 处 理 相关 语句 

MySQL 支持 很 多 与 事务 处 理 相 关 的 语句 ， 通 过 使 用 这 些 语句 可 以 开始 和 终止 事务 ， 设 置 事务 


属性 、 推 迟 约束 执行 、 设 置 断 点 等 。 下 面 将 对 这 些 语句 进行 介绍 。 
(1) START TRANSACTION、COMMIT 和 ROLLBACK 
上 述 语句 用 于 标识 一 个 用 户 定义 的 事务 的 简单 过 程 。 它 们 的 使 用 格式 如 下 所 示 : 
START TRANSACTION | BEGIN [WORK] : 
COMMIT [WORK] [AND [NO] CHAIN] [[NO] RELEASE] “, 
ROLLBACK [WORK] [AND [NO] CHAIN] [[NO] RELEASE] 
SET AUTOCOMMIT = {0 | 1} 


在 上 述 使 用 格式 中 ，START TRANSACTION 或 BEGIN 语句 可 以 开始 一 项 新 的 事务 。COMMIT 
可 以 提交 当前 事务 ， 使 变更 成 为 永久 变更 。ROLLBACK 可 以 回 滚 当前 事务 ， 取 消 其 变更 。SET 


AUTOCOMMIT 语句 可 以 禁用 或 启用 默认 的 autocommit 模式 ， 用 于 当前 连接 。 第 
下 面 对 上 述 使 用 格式 中 的 各 选项 进行 说 明 ， 具 体 如 表 10-9 所 示 。 10 

表 10-9 选项 说 明 

语句 部 分 说 阴 三 
WORK 该 选项 被 支持 ， 用 于 COMMIT 和 RELEASE， 与 CHAIN 和 RELEASE 子 句 。CHAIN 台 
和 RELEASE 可 以 被 用 于 对 事务 完成 进行 附加 控制 。 Completion_type 系统 变量 的 值 决 

定 了 默认 完成 的 性 质 沁 

AND CHAIN 该 子 名 会 在 当前 事务 结束 时 ， 立 刻 启动 一 个 新 事务 ， 并 且 新 事务 与 刚 结束 的 事务 有 相 亩 


同 的 隔离 等 级 。RELEASE 子 句 在 终止 了 当前 事务 后 ， 会 让 服务 器 断 开 与 当前 客户 端 
的 连接 。 包 含 NO 关键 词 可 以 抑制 CHAIN 或 RELEASE 完成 。 如 果 completion_type 
系统 变量 被 设置 为 一 定 的 值 ， 使 连锁 或 释放 完成 可 以 默认 进行 ， 此 时 NO 关键 词 有 用 
autocommit 模式 运行 默认 情况 下 ，MySQL 采用 autocommit 模式 运行 。 也 就 是 说 ， 当 用 户 执行 一 个 用 于 更 
新 (修改 ) 表 的 语句 之 后 ，MySQL 立刻 把 更 新 存储 到 磁盘 中 
禁用 autocommit 模式 如 果 用 户 正在 使 用 一 个 事务 安全 型 的 存储 引擎 (如 InnoDB、BDB 或 NDB 簇 )， 则 可 
以 使 用 以 下 语句 禁用 autocommit 模式 : 


SET AUTOCOMMIT=0; 


通过 把 AUTOCOMMIT 变量 设置 为 零 ， 禁 用 autocommit 模式 之 后 ， 用 户 必须 使 用 
COMMIT 把 变更 存储 到 磁盘 中 ， 如 果 要 忽略 从 事务 开始 进行 以 来 做 出 的 变更 ， 使 用 


ROLLBACK 
使 用 START autocommit 将 被 禁用 ， 直 到 使 用 COMMIT 或 ROLLBACK 结束 事务 为 止 。 然 后 
TRANSACTION autocommit 模式 恢复 到 原来 的 状态 


BEGIN 和 BEGIN WORK | BEGIN 和 BEGIN WORK 被 作为 START TRANSACTION 的 别名 受到 支持 , 用 于 对 事 
务 进行 初始 化 。START TRANSACTION 是 标准 的 SQL 语法 ， 并 且 是 启动 一 个 ad-hoc 
事务 的 推荐 方法 。 BEGIN 语句 与 BEGIN 关键 词 的 使 用 不 同 。 BEGIN 关键 词 可 以 启动 
一 个 BEGIN..END 复合 语句 。 后 者 不 会 开始 一 项 事务 


(2) SAVEPOINT 和 ROLLBACK TO SAVEPOINT 

InnoDB 支持 SQL 语句 SAVEPOINT、ROLLBACK TO SAVEPOINT、RELEASE SAVEPOINT 
和 可 选 的 用 于 ROLLBACK 的 WORK 关键 词 。 它 们 的 使 用 格式 如 下 所 示 : 

SAVEPOINT identifier 


ROLLBACK [WORK] TO SAVEPOINT identifier 
RELEASE SAVEPOINT identifier 


卫 
EE 
卫 
十 

三 
< 

op) 
名 
La 
完 
全 
至 
习 
手 
册 


在 上 述 使 用 格式 中 ，SAVEPOINT 语句 用 于 设置 一 个 事务 保存 点 ， 带 一 个 标识 符 名 称 。 如 果 当 
前 事务 有 一 个 同样 名 称 的 保存 点 ， 则 旧 的 保存 点 被 删除 ， 新 的 保存 点 被 设置 。 

ROLLBACK TO SAVEPOINT 语句 会 向 已 命名 的 保存 点 回 滚 一 个 事务 。 如 果 在 保存 点 被 设置 后 ， 
当前 事务 对 其 进行 了 更 改 ， 则 这 些 更 改 会 在 回 滚 中 被 撤销 。 但 是 ，InnoDB 不 会 释放 被 存储 在 保存 点 
之 后 的 存储 器 中 的 行 锁定 。( 注 意 ， 对 于 新 插入 的 行 ， 锁 定 信息 被 存储 在 行 中 的 事务 ID 承载 ; 锁定 
没有 被 分 开 存储 在 存储 器 中 。 在 这 种 情况 下 ， 行 锁定 在 撤销 中 被 释放 )。 在 被 命名 的 保存 点 之 后 设置 
的 保存 点 被 删除 。 

RELEASE SAVEPOINT 语句 会 从 当前 事务 的 一 组 保存 点 中 删除 已 命名 的 保存 点 。 不 出 现 提交 或 
回 深 。 如 果 保 存 点 不 存在 ， 会 出现 错误 。 如 果 用 户 执 行 COMMIT 或 执行 不 能 命名 保存 点 的 
ROLLBACK， 则 当前 事务 的 所 有 保存 点 被 删除 。 

(3) SET TRANSACTION 

SET TRANSACTION 用 于 设置 事务 隔离 等 级 ， 用 于 下 一 个 事务 ， 或 者 用 于 当前 会 话 。 它 的 使 用 
格式 如 下 所 示 : 


SET [GLOBAL | SESSION] TRANSACTION ISOLRTION LEVEL 
{ READ UNCOMMITTED | READ COMMITTED | REPEATABLE READ | SERIALIZABLE} 


在 默认 情况 下 ，SET TRANSACTION 会 为 下 一 个 事务 (还 未 开始 ) 设置 隔离 等 级 。 如 果 使 用 
GLOBAL 关键 词 ， 则 语句 会 设置 全 局 性 的 默认 事务 等 级 ， 用 于 从 该 点 以 后 创建 的 所 有 新 连接 。 原 有 
的 连接 不 受 影响 。 要 进行 此 操作 ， 用 户 需要 具有 SUPER 权限 。 使 用 SESSION 关键 词 可 以 设置 默认 
事务 等 级 ， 用 于 对 当前 连接 执行 的 所 有 将 来 事务 。 

(4) LOCK TABLES 和 UNLOCK TABLES 

LOCK TABLES 可 以 锁定 用 于 当前 线程 的 表 。 如 果 表 被 其 他 线程 锁定 ， 则 造成 堵塞 ， 直 到 可 以 
获取 所 有 锁定 为 止 。UNLOCK TABLES 可 以 释放 被 当前 线程 保持 的 任何 锁定 。 当 线程 发 布 男 一 个 
LOCK TABLES 时 ， 或 当 与 服务 器 的 连接 被 关闭 时 ， 所 有 由 当前 线程 锁定 的 表 被 隐 含 地 解锁 。 它 们 
的 使 用 格式 如 下 所 示 : 

LOCK TABLES 

tbl name [RS alias] {READ [LOCAL] | [LOW_PRIORITY] WRITE} 


[, tbl name [AS alias] {READ [LOCAL] | [LOW PRIORITY] WRITE}] ... 
UNLOCK TABLES 


表 锁 定 只 用 于 防止 其 他 客户 端 进行 不 正当 的 读 取 和 写 入 。 保 持 锁定 〈 即 使 是 读 取 锁 定 ) 的 客户 
端 可 以 进行 表层 级 的 操作 ， 比 如 DROP TABLE。 

6. 事务 处 理 实例 

下 面 的 事务 处 理 实例 用 于 展示 事务 的 处 理 过 程 ， 从 而 对 事务 及 其 处 理 有 更 深刻 的 认识 。 首 先 ， 
对 数据 表 reply 执行 插入 操作 ， 并 设置 事务 保存 点 sp1; 然后 对 数据 表 reply 执行 更 新 操作 ， 并 设置 
事务 保存 点 ;最 后 显示 出 数据 表 reply 中 的 所 有 数据 。 有 具体 代码 及 执行 情况 如 下 所 示 : 

案例 10-1: 


mysql>START TART TRANSACTION; 
Query OK, 0 rows affected (0.00 sec) 


mysql> INSERT INTO reply (msgid, recontent, retime) VALUES(4,'http://www.itying.net 


RA 
Query OK, 1 row affected (0.05 sec) 


mysql> SAVEPOINT spl; 
Query OK, 0 rows affected (0.00 sec) 


mysql> UPDATE reply SET recontent="Hello SongYan!' WHERE repid=4; 
Query OK, 1 row affected (0.01 sec) 
Rows matched: 1 Changed: 1 Warnings: 0 


mysql> SAVEPOINT sp2; 
Query OK, 0 rows affected (0.00 sec) 


mysql> UPDATE reply SET recontent="Hello MyPHP!' WHERE repid=1; 
Query OK, 1 row affected (0.00 sec) 
Rows matched: 1 Changed: 1 Warnings: 0 


mysql> SELECT * FROM reply; 

成 功 执行 上 述 语句 后 , reply 表 中 的 数据 
如 图 10-26 所 示 。 。 

接着 使 用 ROLLBACK 语句 回 深 到 事务 
保存 点 sp2， 执 行 如 下 所 示 的 语句 : 


mysql> ROLLBACK TO sp2; 
Query OK, 0 rows affected (0.02 sec) 


mysql> SELECT * FROM reply; 


成 功 执行 上 述 语句 后 , reply 表 中 的 数据 
如 图 10-27 所 示 : 

从 图 10-27 可 以 看 出 ， 通 过 回 滚 到 事务 
保存 点 sp2 撤销 所 有 在 该 点 后 所 执行 的 操 
作 。 最 后 执行 COMMIT 语句 进行 操作 的 永 图 10-27 reply 表 中 的 数据 (二 ) 
久 更 改 。 


10.3.2 ”存储 过 程 


MySQL 5.1 及 其 后 续 版 本 提供 了 对 存储 过 程 的 支持 。 一 个 存储 程序 是 可 以 被 存储 在 服务 器 中 的 
一 组 SQL 语句 。 一 旦 它 被 存储 了 ,客户 端 不 需要 再 重新 发 布 单独 的 语句 ,而 是 可 以 引用 存储 过 程 来 
蔡 代 。 

1. MySQL 存储 过 程 概述 

存储 过 程 〈Stored Procedure) 是 数据 库 的 常用 术语 ， 它 支持 SELECT、INSERT、UPDATE 和 
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DELETE 等 SQL 命令 的 执行 ， 还 可 以 设置 能 在 程序 外 引用 的 参数 。 

通常 情况 下 ， 需 要 操作 数据 库 中 的 数据 时 会 使 用 存储 过 程 ， 用 来 获取 记录 、 插 入 、 更 新 或 删除 
值 。 使 用 存储 过 程 具 有 许多 优点 ， 主 要 如 下 所 示 : 

(1) 存储 过 程 与 其 他 应 用 程序 共享 应 用 程序 逻辑 ， 因 而 确保 了 数据 访问 和 修改 的 一 致 性 。 存 储 
过 程 可 以 封装 业务 功能 ， 在 存储 过 程 中 可 以 在 同一 位 置 改变 封装 的 业务 规则 和 策略 。 所 有 的 客户 端 
可 以 使 用 相同 的 存储 过 程 来 确保 数据 访问 和 修改 的 一 致 性 。 

(2) 存储 过 程 具有 安全 性 和 所 有 权 连 接 ， 以 及 可 以 附加 到 它们 的 证 书 。 用 户 可 以 被 授予 权限 来 
执行 存储 过 程 而 不 必 直 接 对 存储 过 程 中 引用 的 对 象 具 有 权限 。 

(3) 存储 过 程 提供 了 安全 机 制 。 即 使 是 没有 访问 存储 过 程 引用 的 表 或 视图 的 权限 的 用 户 ， 也 可 
以 被 授权 执行 该 存储 过 程 。 

(4) 存储 过 程 允许 模块 化 程序 设计 。 存 储 过 程 一 旦 创建 ， 以 后 即 可 在 程序 中 调用 任意 多 次 。 这 
可 以 改进 应 用 程序 的 可 维护 性 ， 并 允许 应 用 程序 统一 访问 数据 库 。 

(5) 存储 过 程 可 以 减少 网 络 通 信 流 量 。 用 户 可 以 通过 发 送 一 个 单独 的 语句 实现 一 个 复杂 的 操作 ， 
而 不 需要 在 网 络 上 发 送 几 百 个 MySQL 代码 ， 这 样 减少 了 在 服务 器 和 客户 机 之 间 传 递 的 请 求 数量 。 

2. 存储 过 程 权限 表 

存储 程序 需要 在 MySQL 数据 库 中 有 proc 表 和 procs_priv 表 。 这 两 个 表 在 MySQL 5.1 安装 过 程 
中 创建 。 如 果 用 户 是 从 早期 的 版 本 升级 到 MySQL 5.1， 需 要 更 新 授权 表 以 确保 proc 表 和 procs_priv 
的 存在 。 

(1) proc 

proc 表 存 储 关 于 存储 过 程 的 信息 , 包括 其 语法 、 创 建 日 期 参数 列表 等 ,具体 结构 信息 如 表 10-10 
所 示 。 

表 10-10 ”proc 表 结构 


字段 默 认 什 
Db char(64) | 是 | 无 默认 值 
name char(64) | 否 | 无 默认 值 
type enumtype 无 默认 什 
specific_name char(64) 无 默认 值 
language enmu(SQL) | | so 
sql_data_ access enumdataaccess CONTAINS SQL 
is_deterministic enum(CYES NO) 否 NO 
Sectrity type enumsecurity 否 DEFINER 
param list blob 否 无 默认 值 
returns char(64) 否 无 默认 值 
body longblob 否 无 默认 值 
definer char (77) 否 无 默认 值 
created timestamp 是 当前 时 间 惟 
modified timestamp 0000-00-00 00:00:00 
sql mode setsqlmode 无 默认 值 
comment char(64) -一 一 一 无 默认 值 


在 上 述 表 中 ，enumtype 用 来 表示 enum(FUNCTION',PROCEDURE),， 它 确定 此 例 程 是 函数 还 是 


过 程 .enumdataaccess 用 来 表示 enum(CONTAINS SQL'NO_SQL'READS SQL DATA'MODIEFIES 
SQL DATA))。 

(2) procs priv 

procs_priv 表 用 于 存储 相关 的 权限 信息 ， 这 些 信息 反映 哪些 用 户 允 许 与 定义 在 proc 表 中 的 例 程 
交互 。 具 体 结构 信息 如 表 10-11 所 示 。 


表 10-11 procs_priv 表 结构 


字 段 默 认 值 
host 无 默认 值 
db 无 默认 值 
User char(16) 无 默认 值 
routine name char(64) 无 默认 值 
routine type enumroutine 无 默认 值 
grantor 无 默认 值 
proc_priv 无 默认 值 
timestamp 当前 时 间 戳 


在 上 述 表 中 , enumroutine 用 来 表示 enum(FUNCTION',PROCEDURE), 它 确定 此 例 程 是 函数 还 
是 过 程 。 其 实 这 个 信息 在 表 proc 中 己 经 有 重复 的 出 现 , 之 所 以 重复 此 信息 是 因为 存储 过 程 和 存储 函 
数 可 能 同名 ， 所 以 MySQL 需要 此 信息 来 确定 指定 用 户 可 以 查看 哪个 例 程 。 

proset 用 来 表示 set(Execute','Alter Routine','Grant")。 对 执行 指定 例 程 的 用 户 必须 设置 为 Execute， 
对 于 修改 或 删除 此 例 程 的 用 户 必须 设置 为 Alter Routine， 如 果 用 户 要 将 此 例 授权 给 其 他 用 户 ， 必 须 
设置 为 Grant。 

3. 存储 过 程 变量 

与 PHP 不 同 ， 在 存储 过 程 使 用 局 部 变量 之 前 必须 声明 局 部 变量 ， 通 过 使 用 MySQL 支持 的 某 种 
数据 类 型 指定 变量 类 型 。 变 量 声明 使 用 DECLARE 语句 实现 ， 该 语句 的 具体 使 用 格式 如 下 所 示 : 

DECLARE var name[,...] type [DEFAULT value] 

该 语句 用 来 声明 局 部 变量 。 要 给 变量 提供 一 个 默认 值 ， 需 要 包含 一 个 DEFAULT 子 句 。 值 可 以 
被 指定 为 一 个 常数 或 一 个 表达 式 。 如 果 没 有 DEFAULT 子 句 ， 初 始 值 为 NULL。 局 部 变量 的 作用 范 


围 在 它 被 声明 的 BEGIN .… END 块 内 。 它 可 以 被 用 在 没有 相同 名 称 变量 的 嵌 套 块 中 。 声 明 变量 的 示 
例如 下 所 示 : 


DECLARE username VARCHAR(20); 
DECLARE total DECIMAL(6,2); 


声明 了 变量 以 后 就 可 以 对 此 变量 设置 相应 的 值 了 。SET 语句 就 是 用 来 设置 声明 的 存储 过 程 变量 
值 的 。 它 的 使 用 格式 如 下 所 示 : 


SET var name = expr [, Var name = expr] ... 


下 面 就 用 SET 语句 声明 并 设置 变量 stuname， 具 体 如 下 所 示 : 


DECLARE stuname VARCHAR(20); 
SET stuname = ' 宋 岩 岩 '; 
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用 户 也 可 以 使 用 SELECT INTO 语句 设置 变量 。 关 于 它 的 具体 使 用 可 以 参考 下 一 小 节 中 的 相关 
内 容 。 
4. 创建 存储 过 程 
创建 存储 过 程 的 具体 使 用 格式 如 下 所 示 : 


CREATE PROCEDURE sp name ([proc parameter[,...]]) 
[characteristic ...] routine body 


在 上 述 使 用 格式 中 ， 由 括号 包围 的 存储 过 程 参数 列 必 须 总 是 存在 。 如 果 没 有 参数 ， 应 该 使 用 一 
个 空 参数 列 0。 每 个 参数 默认 都 是 一 个 IN 参数 。 要 指定 为 其 他 参数 ， 可 在 参数 名 之 前 使 用 关键 词 
OUT 或 INOUT。 其 中 ，IN 参数 用 来 向 过 程 传递 信息 ; OUT 参数 用 来 从 过 程 传 回信 息 ; INOUT 参 
数 用 来 向 过 程 传递 信息 ， 如 果 值 改变 ， 则 可 以 在 过 程 外 进行 调用 。 注 意 ， 对 于 任何 声明 为 OUT 或 
INOUT 的 参数 ， 当 调用 存储 过 程 时 需要 在 参数 名 前 加 上 @ 符 号 。routine_body 包含 合法 的 SQL 过 程 
语句 。 可 以 使 用 复合 语句 语法 ， 并 且 这 些 复合 语句 可 以 包含 声明 、 循 环 和 其 他 控制 结构 语句 。 

下 面 创建 一 个 存储 过 程 ShowAll， 用 于 显示 reply 表 中 的 所 有 信息 ， 包 括 字段 repid、msgid、 
recontent 和 retime。 具 体 语 句 如 下 所 示 : 


USE guestbook 
CREATE PROCEDURE ShowAll() 
SELECT * FROM reply; 


下 面 就 可 以 使 用 CALL 命令 执行 存储 过 程 ShowAll0 了， 具体 语句 如 下 所 示 : 
CALL ShowAll(); 


成 功 执行 上 述 语句 将 输出 如 图 10-28 所 示 的 结果 。 

上 面 示例 是 比较 简单 的 存储 过 程 使 用 。 
在 通常 情况 下 ， 存 储 过 程 通过 参数 与 调用 它 
的 程序 通信 。 也 就 是 说 ， 在 程序 调用 存储 过 
程 时 ， 可 以 通过 输入 参数 将 数据 传 给 存储 过 
程 ， 存 储 过 程 可 以 通过 输出 参数 和 返回 值 将 
数据 返回 给 调用 它 的 程序 。 

下 面 是 一 个 使 用 IN、OUT 参数 的 存储 
过 程 示例 。 该 示例 在 程序 被 定义 时 ， 用 
MySQL 客户 端 delimiter 命令 来 把 语句 定 界 
符 从 “;” 变 为 “/”。 这 就 允许 用 在 程序 体 中 
的 “;” 定 界 符 被 传递 到 服务 器 而 不 是 被 图 10-28 ”调用 存储 过 程 的 结果 
MySQL 自己 来 解释 。 有 具体 过 程 如 下 所 示 : 

(1) 修改 定 界 符 为 “//”， 创建 存 储 过 程 ShowOne0， 有 具体 语句 如 下 所 示 : 


mysql> DELIMITER // 
mysql> CREATE PROCEDURE ShowOne(IN rid INT,OUT result INT) 
-> BEGIN 


-> SELECT COUNT (+) INTO result FROM reply WHERE repid=rid; 


-> END 

= 
Query OK, 0 rows affected (0.00 sec) 2 
(2) 修改 定 界 符 为 “;”， 调 用 存储 过 程 ShowOne0， 并 输出 执行 结果 ， 有 具体 语句 如 下 所 示 : ee 


mysql> DELIMITER ; 
mysql> CALL ShowOne(5,@result); 
Query OK, 0 rows affected (0.00 sec) 


mysql> SELECT eresult7 


上 述 示例 的 执行 过 程 及 结果 如 图 10-29 
所 示 。 

5. 管理 存储 过 程 

在 MySQL 中 , 用 户 可 以 通过 ALTER 语 
名 修改 存储 过 程 。 具 体 使 用 格式 如 下 所 示 : 


ALTER PROCEDURE sp_name 
[characteristic ...] 
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注意 ， 在 MySQL 5.1 中 ， 用 户 必 须 具 有 
ALTER ROUTINE 权限 才 可 以 使 用 此 子 程 
序 。 这 个 权限 被 自动 授予 子 程序 的 创建 者 。 10-29 使 用 IN、OUT 参数 的 存储 过 程 示 例 
例如 ， 如 果 希 望 修改 ShowAll0 方 法 的 SQL 
SECURITY 特点 ， 将 其 从 默认 的 DEFINER 改 为 INVOKER。 具 体 语 句 如 下 所 示 : 


ALTER PROCEDURE ShowR11 SQL SECURITY invokey 

另外 ， 如 果 某 一 存储 过 程 不 再 需要 ， 可 以 执行 DROP 语句 将 其 删除 。 有 具体 使 用 格式 如 下 所 示 : 

DROP PROCEDURE [IF EXISTS] sp_name 

同 ALTER 语句 一 样 ， 用 户 要 具有 执行 该 语句 的 权限 才 行 。 例 如 ， 删 除 ShowAlO 存 储 过 程 的 具 
体 语句 如 下 所 示 : 

DROP PROCEDURE ShowAll; 

6. 查看 存储 过 程 

用 户 可 以 使 用 SHOW STATUS 语句 查看 某 个 存储 过 程 的 所 有 者 、 数 据 库 、 名 字 、 类 型 、 创 建 时 
间或 修改 时 间 等 相关 信息 。 该 语句 的 具体 使 用 格式 如 下 所 示 : 


SHOW PROCEDURE STATUS [LIKE "pattern'"] 


在 使 用 上 述 语句 时 ， 如 果 没有 指定 样式 ， 将 根据 使 用 的 语句 ， 所 有 存储 程序 的 信息 都 被 列 出 。 
例如 ， 如 果 希 望 查看 前 面 创建 的 ShowOneO 存 储 过 程 的 信息 。 具 体 语 句 及 执行 结果 如 下 所 示 : 
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mysql> SHOW PROCEDURE STATUS LIKE "ShowOne'\G 
训 间 束 于 于 于 于 于 于 兴 束 于 束 呈 于 汪 字 本 于 宙 字 于 间 本 可 站] 。 工 O 〇 WW 本 地 可 可 本 可 本 可 宙 可 可 可 李 可 可 宙 可 本 机 宙 束 可 可 宁可 宙 


Db: guestbook 
Name: ShowOne 
Type: PROCEDURE 
Definer: root@localhost 
Modified: 2007-06-29 10:06:35 
Created: 2007-06-29 10:06:35 
Security type: DEFINER 
Comment: 
1 row in set (0.05 sec) 


注意 ， 在 上 述 语句 中 ， 使 用 \G 选项 是 为 了 以 垂直 格式 而 不 是 水 平 格式 显示 输出 。 不 包括 \G 将 
水 平地 显示 结果 。 

此 外 , 用 户 可 以 通过 SHOW CREATE 语句 查看 创建 特定 存储 过 程 所 用 的 语法 。 该 语句 的 具体 使 
用 格式 如 下 所 示 : 

SHOW CREATE PROCEDURE sp name 

例如 ， 如 下 所 示 的 语句 将 重新 创建 用 于 
创建 ShowOneO 存 储 过 程 的 语法 : 

mysql> SHOW CREATE PROCEDURE 

Showone \G; 


成 功 执行 上 述 语句 将 输出 如 图 10-30 所 
示 的 结果 。 图 10-30 查看 创建 存储 过 程 所 用 的 语法 


10.4 使 用 MySQL inistrator 管理 数据 库 


MySQL Administrator 是 MySQL AB 的 产品 ， 它 是 一 种 专门 用 于 管理 MySQL 4.X 和 5.X 的 数据 
库 软 件 。 它 提供 了 众多 的 功能 ， 例 如 ， 创 建 表 、 编 辑 表 、 备 份 MySQL 数据 库 ， 还 原 MySQL 数据 
库 等 。MySQL Administrator 在 2004 年 1 月 公开 发 行 ， 并 采用 
双 许 可 模型 发 行 ， 即 GNU 通用 公开 许可 (GPL) 和 商业 许可 。 
Linux、Microsoft 和 Mac OS X 平台 都 有 相应 的 版 本 ， 并 提供 二 
进 制 包 、RPM 和 源 代码 包 。 无 论 选择 何 种 许可 ， 都 是 免费 的 。 
户 可 以 从 官方 网 站 下 载 MySQL Administrator 的 二 进 制 
包 ， 并 进行 安装 ， 安 装 完成 后 启动 MySQL Administrator， 会 显 
示 如 图 10-31 所 示 的 登录 对 话 框 。 
户 按 照 安装 MySQL 时 的 配置 ， 输 入 相应 的 登录 信息 ， [res] Ca] Cow Cam 
类 似 于 图 10-31。 如 果 登 录 成 功 ， 会 显示 如 图 10-32 所 示 的 
MySQL Administrator 窗口 。 


图 10-31 登录 对 话 框 
这 样 用 户 就 可 以 使 用 MySQL Administrator 对 MySQL 数据 库 进行 管理 了 。 它 提供 了 众多 操作 


MySQL 数据 库 的 功能 ， 具 有 以 下 一 些 显著 特性 : 
(1) 管理 服务 器 各 个 方面 的 接口 ， 包 括 守 护 进程 服务 、 用 户 和 权限 、 配 置 变量 和 日 志 以 及 基于 
GUI 方式 的 数据 库 备 份 与 恢复 等 。 数 据 库 备 份 的 窗口 如 图 10-33 所 示 。 
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图 10-32 MySQL Administrator 窗口 
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10-33 ”数据 库 备 份 窗口 


(2) 实时 图 形 化 地 监视 连接 和 内 存 使 用 、 流 量 、SQL 查询 、 复 制 状态 和 用 户 连 接 。 
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(3) 详细 的 用 户 管理 ， 人 允许 管理 员 管理 每 个 用 户 的 用 户 名 、 密 码 、 权 限 和 资源 使 用 。 此 外 ， 管 
理 员 可 以 维护 每 个 用 户 的 联系 信息 ， 包 括 姓名 、 电 子 邮 件 地 址 、 描 述 、 其 他 联系 信息 ， 甚 至 是 照片 。 


10.5 ”使 用 phppMyAdmin 管理 数据 库 


phpMyAdmin 是 基于 Web 的 MySQL 管理 应 用 程序 ， 使 用 PHP 编写 。 它 不 仅 非 常 稳定 , 而 且 具 
有 丰富 的 特性 ， 具 体 如 下 所 示 : 

(1) phpMyAdmin 是 基于 浏览 器 的 ， 使 用 户 从 任何 能 够 访问 Web 的 地 方 都 能 轻松 地 管理 远程 
MySQL 数据 库 。SSL 也 得 到 了 透明 的 支持 ， 如 果 用 户 的 服务 器 提供 了 这 个 特性 ， 就 能 实现 加 密 的 
管理 。 

(2) 管理 员 可 以 完全 控制 用 户 权 限 、 密 码 和 资源 使 用 ， 还 可 以 创建 、 删 除 甚至 复制 用 户 账户 。 

(3) 实时 界面 可 以 查看 正常 运行 的 信息 、 查 询 和 服务 器 流量 统计 信息 、 服 务 器 变量 和 正在 运行 
的 进程 。 

phpMyAdmin 的 界面 被 全 世界 的 开发 者 翻译 为 49 种 语言 ， 包 括 中 文 (繁体 和 简体 )、 英 语 、 法 
语 、 阿 拉 伯 语 等 。 

(4) phpMyAdmin 提供 了 优化 的 单 击 界面 ， 大 大 减少 了 用 户 出 错 的 可 能 性 。 

另外 ，phpMyAdmin 遵循 GNU 通用 公开 许可 发 行 。 用 户 可 以 从 phpMyAdmin 官方 网 站 http:// 
www.phpmyadmin.net 下 载 源 代 码 ， 并 可 以 查看 新 闻 、 邮 件 列表 及 在 线 演示 等 有 关 phpMyAdmin 的 
相关 信息 。 

下 面 介绍 一 下 如 何 配置 phbpMyAdmin， 以 便于 管理 MySQL 数据 库 。 有 具体 过 程 如 下 所 示 : 

(1) 首先 要 获得 phpMyAdmin 的 压缩 文档 ， 可 以 从 官方 网 站 直接 下 载 ， 这 里 获得 的 版 本 
是 phpMyAdmin 2.10.1。 然 后 将 解压 后 的 所 有 文件 复制 到 C:\Apache2.2\htdocs\phpMyAdmin 子 目 
录 中 。 

(2) 编辑 C:\Apache2.2\htdocs\phpMyAdmin\libraries 目录 中 的 config.default.php 文件 ， 主 要 修改 
如 下 所 示 的 几 项 设置 : 

$cfg['PmaAbsoluteUri'] = 'http://localhost/phpmyadmin'; 

$cfg['Servers'] [$i]['user'] = "root'7 

$cfg['Servers'] [$i] ['password'] = "123456789'; 


上 述 第 一 行 用 户 要 根据 自己 当初 配置 PHP 环境 时 的 设置 进行 设 定 。 而 第 二 、 三 行 是 访问 MySQL 
数据 库 的 用 户 名 和 密码 ， 这 两 项 设置 用 户 也 要 根据 自己 当初 配置 MySQL 时 的 设置 进行 设 定 。 

(3) 重新 启动 Apache 服务 器 ， 打 开 正 浏览 器 在 地 址 栏 中 输入 如 下 所 示 的 地 址 : 

http://localhost/phpmyadmin/index.php 

如 果 执 行 成 功 ， 会 显示 phpMyAdmin 的 初始 页 面 ， 如 图 10-34 所 示 。 
由 于 phpMyAdmin 通过 Web 形式 提供 了 可 视 化 的 操作 环境 ， 所 以 用 户 可 以 很 容易 地 掌握 它 的 
使 用 ， 通 过 它 来 创建 或 管理 用 户 、 数 据 库 、 表 ， 导 入 、 导 出 数据 ， 以 及 执行 SQL 语句 等 大 部 分 的 
MySQL 功能 。 
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图 10-34 _ phpMyAdmin 的 初始 页 面 
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第 11 章 PHP 和 数据 访问 
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PHP 动态 网 站 技术 ， 其 中 一 个 重要 特征 就 是 数据 的 交互 ， 既 可 以 从 客户 端 向 MySQL 服务 器 端 
插入 数据 ， 也 可 以 从 服务 器 端 获 取 并 显示 数据 。 完 成 这 样 的 操作 ， 需 要 使 用 PHP 提供 的 MySQL 函 
数 库 ,同样 也 可 以 使 用 PHP 中 的 扩展 函数 库 MySQLI 完成 。 本 章 将 从 这 两 个 函数 库 入 手 , 介绍 PHP 
连接 数据 库 ， 执 行 查询 语句 、 准 备 语句 、 事 务 处 理 等 知识 点 ， 并 在 本 章 的 最 后 介绍 使 用 ODBC 数据 
源 连 接 Access 数据 库 。 


FY 内 容 摘要 | avstract 


掌握 在 PHP 中 启用 MySQL 和 MySQLI 函数 库 
掌握 连接 数据 库 的 步骤 

掌握 在 PHP 中 数据 库 的 基本 操作 

掌握 在 PHP 中 获取 和 显示 数据 

掌握 在 PHP 获取 表 和 字段 的 信息 

掌握 在 PHP 中 获取 报错 消息 

掌握 在 PHP 中 使 用 MySQLI 函数 库 

掌握 PHP 的 查询 及 多 个 查询 

掌握 PHP 的 准备 语句 和 事务 处 理 


11.1 ”准备 工作 


从 前 面 的 章节 可 以 了 解 到 , PHP 对 每 一 个 对 象 的 操作 实际 上 都 是 通过 函数 来 完成 的 , 如 字符 串 、 
数组 、 文 件 等 。 同样， PHP 对 MySQL 数据 库 的 操作 也 是 通过 函数 来 完成 的 。 在 使 用 字符 串 函 数 时 ， 
直接 调用 这 些 函 数 就 可 以 了 ,不 需要 什么 设置 ,因为 这 些 函 数 都 是 内 置 的 ,在 PHP 5.0 中 ,对 于 MySQL 
数据 库 操作 的 函数 不 是 内 置 函 数 ， 而 属于 第 三 方 函数 ， 这 些 函 数 存在 于 php_mysql.dll 库 中 。 故 在 使 
用 这 些 函 数 之 前 ， 需 要 把 这 些 函 数 所 在 的 库 引入 进来 。 

如 果 要 引入 php_mysql.dll 函数 库 ， 需 要 在 php.ini 文件 中 进行 设置 。 打 开 PHP 的 安装 目录 
Ci\Web\php 文件 夹 ， 找 到 php ini 文件 ， 打 开 文 件 。 在 该 文件 中 ， 从 第 589 行 起 有 如 下 的 语句 块 : 


Vvvveyvyyv 


vvy 


;? Windows Extensions 

;? Note that ODBC support is built in, so no dll is needed for it. 

; Note that many DLL files are located in the extensions/ (PHP 4) ext/ (PHP 5) 
; extension folders as well as the separate PECL DLL download (PHP 5). 

;? Be sure to appropriately set the extension dir directive. 


?extension=php bz2.dl1 


| dL 
extension=/ext/php_mysqli.dll // 该 库 为 PHP 扩 展 库 


?extension=php zip.dl1 


该 语句 块 表示 在 PHP 中 需要 引入 的 Windows 扩展 包 。 这 里 可 以 根据 需要 ， 把 PHP 程序 中 需要 
的 包 引 入 进来 。 这 些 包 在 PHP 5.0 中 ， 默 认 情况 下 都 是 禁用 的 。 要 启用 MySQL 函数 的 支持 ， 应 取 
消 php.ini 文件 中 的 注释 ， 即 去 掉 该 行 前 面 的 分 号 ， 并 指明 这 些 类 库 所 在 的 路 径 。 第 三 方 的 类 库 文件 
通常 保存 在 PHP 的 安装 目录 C:\Web\php\ext 文件 夹 下 ， 在 要 启用 的 类 库 前 加 入 库 所 在 的 路 径 。 在 
php.ini 中 ， 启 用 MySQL 函数 的 情形 如 上 述 代码 所 示 。 
设置 好 php.ini 之 后 ， 就 可 以 重新 启动 Apache 服务 器 ， 如 果 服 务 器 启动 成 功 ， 就 需要 等 待 下 一 
步 检 测 ， 否则 ， 就 表示 配置 有 问题 。 
在 进行 PHP 和 MySQL 数据 库 连 接 之 前 ， 还 需要 创建 一 个 MySQL 数据 库 作 为 连接 使 用 。 局 动 
MySQL 数据 库 ， 在 客户 端 输入 下 列 信息 : 
create database com; 
use com; 
create table cp( 
cp id int not null auto increment, 
cp name varchar (25) not null, 
cp_price decimal(5,2) not null, 
cp_des mediumtext not null, 
primary key(cp_id) 
); 
数据 库 和 表 创建 好 后 ， 就 可 以 进行 数据 库 连 接 了 。 无 论 用 户 何 时 连接 MySQL 服务 器 ， 都 需要 
进行 用 户 权限 的 验证 。 这 里 连接 MySQL 服务 器 用 户 为 root， 密 码 为 root。 


11.2 ”连接 MySQL 数据 库 


现在 可 以 在 PHP 页 面 连接 数据 库 了 ， 这 里 可 以 把 PHP 脚本 程序 看 作 一 个 对 象 ， 存 放 数据 的 
MySQL 数据 库 看 作 一 个 对 象 ， 两 个 对 象 如 果 要 进行 对 话 ， 二 者 之 间 应 该 存在 一 个 连接 桥梁 。 建 立 
好 连接 之 后 ， 就 可 以 进行 数据 库 的 各 项 操作 ， 如 从 数据 库 中 获取 信息 ， 并 到 PHP 页 面 显示 或 从 PHP 
页 面向 MySQL 发 送 SQL 语句 等 。 本 节 将 介绍 PHP 脚本 程序 连接 MySQL 数据 库 。 


11.2.1 ”建立 连接 


PHP 连接 数据 库 可 以 有 两 种 方式 ， 一 种 是 通过 PHP 的 MySQL 相关 函数 ， 男 一 种 是 通过 PHP 
的 ODBC 相关 函数 。 本 节 主 要 介绍 PHP 的 MySQL 相关 函数 。 
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在 PHP 中 ， 连 接 数据 库 需 要 使 用 函数 mysql connect0、mysql pconectO、mysql_close0。 前 两 
个 函数 表示 创建 连接 ， 最 后 一 个 函数 表示 关闭 连接 。 其 详细 信息 如 表 11-1 所 示 。 
表 11-1 连接 MySQL 常 用 函数 
语法 格式 


Tesource mysql connect([string $server [. string $usermame 
[. string $password [. bool $new_link [. int $client fags]]]]]) 
Tesource mysql_ pconnect([string $server [, string $usemame 
[. string $password [. int $client fags]]]] ) 

bool mysql close ( [resource $link identifier] ) 


名 称 


mysql_connectO 


功 能 
打开 一 个 到 MySQL 服务 器 
的 连接 

打开 一 个 到 MySQL 服务 器 
的 持久 连接 

关闭 MySQL 连接 


Imysql pconnectO 


Inysql closeO 


1. mysql_connect() 

mysql_connect0 函 数 的 主要 作用 是 打开 或 重复 使 用 一 个 到 MySQL 服务 器 的 连接 。 其 中 ， 参 数 
server 表示 要 连接 的 MySQL 服务 器 。 可 以 包括 端口 号 ， 例 如 “hostname:port”， 或 者 到 本 地 套 接 字 
的 路 径 ， 例 如 对 于 localhost 的 “:/path/to/socket”。 如 果 PHP 指令 mysql.default host 未 定义 〈 默 认 情 
况 )， 则 默认 值 是 “localhost:3306” 或 者 使 用 localhost。 

参数 username 表示 访问 MySQL 数据 库 的 用 户 名 ， 默 认 值 是 服务 器 进程 所 有 者 root 的 用 户 名 。 
参数 password 表示 用 户 具有 的 密码 ， 默 认 值 是 空 密码 。 

参数 new_link 表示 如 果 用 同样 的 参数 第 二 次 调用 mysql_connectO 函 数 ， 将 不 会 建立 新 连接 ， 而 
将 返回 已 经 打开 的 连接 标识 。 可 以 通过 设置 参数 new_link 来 改变 此 行为 ， 并 使 mysql_connectO 函 数 
总 是 打开 新 的 连接 ， 即 使 mysql_connect0 函 数 曾 在 前 面 被 用 同样 的 参数 调用 过 。 

client_flags 参数 可 以 是 以 下 常量 的 组 合 : MYSQL _CLIENT SSL、MYSQL CLIENT_COMP- 
RESS、MYSQL CLIENT IGNORE_SPACE 或 MYSQL _ CLIENT INTERACTIVE。 其 详细 信息 如 表 
11-2 所 示 。 


表 11-2 client_flags 属 性 表 


选 项 说 明 
MYSQL CLIENT COMPRESS 使 用 压缩 的 通信 协议 
MYSQL CLIENT IGNORE SPACE 允许 在 函数 名 后 留 空 格 位 


MYSQL CLIENT INTERACTIVE 人 允许 设置 断 开 连接 之 前 所 空闲 等 候 的 interactive_ timeout 
时 间 〈 代 替 wait timeout) 

使 用 SSL 加 密 。 该 标志 仅 在 MySQL 客户 端 库 版 本 为 4X 
或 更 高 版 本 时 可 用 。 在 PHP 4.0 和 Windows 版 的 PHP 5.0 


安装 包 中 绑 定 的 都 是 3.23.X 


MYSQL CLIENT SSL 


如 果 函 数 执行 成 功 ， 就 会 返回 一 个 MySQL 连接 标识 ， 否 则 返回 False。 

下 面 创 建 一 个 案例 ， 演 示 利 用 mysql_connect0O 函 数 连 接 MySQL 数据 库 。 在 目录 C:\Web\ 
apache\htdocs 文件 夹 中 ， 建 立 一 个 Data 文件 夹 用 来 存放 关于 数据 库 操作 的 文件 ， 本 章 中 的 文件 如 果 
不 特别 声明 都 是 放 在 该 文件 夹 中 。 在 Data 文件 中 ， 打 开 一 个 记事 本 ， 输 入 下 列 信 息 : 

案例 11-1 

<2php 


$link=mysql] connect ("localhost","root", "root"); 


if(!$1link) 
echo "数据 库 连接 失败 "; 


else 


echo "数据 库 连接 成 功 "; 
} 
mysql close(); 
上 


将 该 文件 保存 ， 文 件 名 为 conn.php。 打 开 正 浏览 器 ， 在 地 址 栏 中 输入 http://localhost:8080/ 
Data/conn.php， 单 击 【 转 到 】 按 钮 ， 会 显示 如 图 11-1 所 
示 的 窗口 。 

在 上 述 代 码 中 ， 调 用 函数 mysql_connect0 进 行 数据 
库 连 接 ， 并 在 函数 中 提供 了 三 个 参数 ， 第 一 个 localhost 
表示 要 连接 的 MySQL 服务 器 ， 即 在 本 机 上 ; 后 面 两 个 
分 别 表示 用 户 名 和 用 户 密码 ， 其 值 都 为 root。 如 果 出 现 图 11-1 连接 数据 库 
了 上 面 窗口 中 的 信息 , 表示 数据 库 连 接 成 功 ,如果 出 现 的 信息 是 “Fatal error: Call to undefined function 
mysql_connect() in http://localhost:8080/Data/conn.php on line 2”， 则 表示 函数 库 php_mysql.dll 没有 被 
正确 地 引入 进来 。 

mysql_connect 除了 上 面 的 连接 方式 外 ， 还 可 以 使 用 下 面 几 种 形式 : 


$link = mysql connect('example.com:3307', ‘mysql user', 'mysql password') 7 

$link = mysql connect('127.0.0.1:3307', 'mysql user', 'mysql password'); 

$link = mysql_ connect('/tmp/mysql', 'mysql user', 'mysql password'); 

$link = mysql_ connect('localhost:/tmp/mysql.sock', "mysql_user'v 

"mysql password') 7 

2. mysql_pconnect() 

Imysql pconnectO 函 数 如 果 执 行 成 功 , 则 返回 一 个 正 的 MySQL 持久 连接 标识 符 , 否则 返回 False。 
Inysql pconnectO 函 数 主要 建立 一 个 到 MySQL 服务 器 的 连接 ， 如 果 没 有 提供 可 选 参数 ， 则 使 用 如 下 
默认 值 : server 为 “localhost:3306”，username 为 服务 器 进程 所 有 者 的 用 户 名 ，password 为 空 密码 。 
client flags 参数 可 以 是 以 下 常量 的 组 合 : MYSQL CLIENT COMPRESS、MYSQL CLIENT_ 
IGNORE_SPACE 或 者 MYSQL_CLIENT INTERACTIVE, 这 些 属性 的 含义 同 mysql_connect0 函 数 的 
属性 相同 。 

mysql_pconnect0 函 数 和 mysql_connect0 函 数 功能 非常 相似 ,一 个 返回 普通 连接 ， 一 个 返回 持久 
连接 。 但 有 两 个 主要 区 别 ， 首 先 ， 当 连接 时 该 函数 将 先 尝试 寻找 一 个 在 同一 个 主机 上 用 同样 的 用 户 
名 和 密码 已 经 打开 的 (持久 ) 连接 ， 如 果 找到 ， 则 返回 此 连接 标识 而 不 打开 新 连接 。 其 次 ， 当 脚本 
执行 完毕 后 到 MySQL 服务 器 的 连接 不 会 被 关闭 ， 此 连接 将 保持 打开 以 备 以 后 使 用 (mysql_closeO 
函数 不 会 关闭 由 mysql_pconnect0 函 数 建立 的 连接 )。 该 函数 的 使 用 示例 如 下 所 示 : 

<?php 


$link=mysql pconnect ("localhost","root", "root"); 
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if(!$link) 
i 
echo "数据 库 连接 失败 "; 


echo "数据 库 连 接 成 功 "; 

?> 

3. mysql_close() 

Imysql_closeO 函 数 表示 关闭 指定 的 连接 标识 所 关联 的 到 MySQL 服务 器 的 非 持久 连接 .如果 没有 
指定 link identifier， 则 关闭 上 一 个 打开 的 连接 。 通 常 不 需要 使 用 mysql_close0 函 数 ， 因 为 已 打开 的 
非 持久 连接 会 在 脚本 执行 完毕 后 自动 关闭 。 

参数 link identifier 表示 连接 标识 符 。 如 果 没 有 指定 ， 默 认 关闭 最 后 被 mysql_connectO 函 数 打 开 
的 连接 。 如 果 没 有 找到 该 指定 连接 ， 函 数 会 尝试 调用 mysql_connectO 函 数 建立 一 个 连接 并 使 用 它 。 
如 果 发 生意 外 ， 没 有 找到 连接 或 无 法 建立 连接 ， 系 统 会 发 出 E_WARNING 级 别 的 警告 信息 。 

如 果 该 函数 能 成 功 执行 ， 则 返回 一 个 Tue， 和 否则 返回 False。 


11.2.2 单独 存放 连接 文件 


在 一 个 大 型 的 Web 程序 中 ， 可 能 会 多 次 使 用 到 连接 数据 库 的 功能 ， 或 数据 库 的 某 个 特定 操作 ， 
那么 这 时 再 在 Web 页 面 中 ， 多 次 使 用 创建 数据 库 连 接 的 代码 ,就 有 点 重复 了 。 其 中 的 一 个 解决 办 法 
是 ， 把 数据 库 连 接 的 代码 放 到 一 个 单独 的 文件 中 ， 其 他 的 文件 如 果 涉 及 到 这 个 功能 可 以 直接 调用 该 
文件 。 这 样 做 的 好 处 是 方便 数据 库 的 代码 修改 ， 并 达到 功能 的 相互 分 离 和 代码 重用 的 目的 。 

把 连接 数据 库 的 代码 放 到 一 个 独立 的 PHP 文件 中 ， 其 代码 如 下 所 示 : 

<?php 


$1link=mysql_connect ("localhost", "root", "root"); 
?> 


将 该 文件 保存 ， 名 称 为 cl.php。 如 果 一 个 PHP 页 面 使 用 到 连接 数据 库 的 代码 ， 可 以 通过 下 列 代 
码 调用 : 


<?php 

1 once ("c1.php") 7 

县 六 

同样 ， 也 可 以 把 连接 数据 库 的 代码 做 成 函数 ， 或 者 一 个 类 。 
11.2.3 ”选择 数据 库 


当 在 PHP 页 面 程序 和 MySQL 服务 器 两 个 对 象 之 间 建 立 一 个 连接 后 , 就 可 以 在 PHP 页 面 的 脚本 


程序 中 对 MySQL 服务 器 中 的 数据 库 进行 操作 。 选择 对 哪个 数据 库 操作 ， 需 要 通过 mysql_select db0 
mysql_select_db0 函 数 的 语法 格式 如 下 所 示 : . 
bool mysql select db ( string $database name [, resource $ link identifier ] ) : 
mysql select db0 函 数 主要 设 定 与 指定 的 连接 标识 符 关 联 的 服务 器 上 的 当前 激活 数据 库 。 如 果 没 1 

有 指定 连接 标识 符 ， 则 使 用 上 一 个 打开 的 连接 。 如 果 没 有 打开 的 连接 ， 该 函数 将 无 参数 调用 

mysql_connect0 函 数 来 尝试 打开 一 个 连接 并 使 用 之 。 该 函数 的 使 用 示例 如 下 所 示 : 


<?php 
$link=mysql connect ("localhost", "root", "root"); 第 
if(!$1link) 
1 11 
echo "数据 库 连接 失败 "; 
入 
elge 


echo "数据 库 连接 成 功 "; 
} 
mysql_ select db("com"); 
mysql_close(); 
?> 
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在 上 述 代码 中 ，com 表示 在 MySQL 服务 器 中 创建 的 数据 库 。 


11.3 ”数据 库 基本 操作 


通过 Web 程序 的 客户 端 ， 可 以 执行 数据 记录 的 添加 、 删 除 、 修 改 或 者 显示 等 操作 ， 从 而 达到 数 
据 更 新 的 目的 。 本 节 将 详细 地 介绍 在 PHP 中 对 MySQL 数据 库 的 基本 操作 。 


11.3.1 执行 SQL 语句 


动态 网 站 的 交互 , 无 论 做 任何 的 操作 其 形式 都 是 一 样 的 , 即 把 需要 执行 的 操作 通过 SQL 语句 表 
达 出 来 ， 然 后 通过 PHP 程序 传递 给 MySQL 数据 库 管理 系统 ， 在 MySQL 中 执行 完毕 后 返回 处 理 结 
果 。 在 PHP 中, 完成 执行 SQL 语句 的 两 个 函数 分 别 是 mysql_queryO 和 mysql db_query0。 其 详细 信 
息 如 表 11-3 所 示 。 
表 11-3 ”执行 SQL 语句 函数 
名 称 语法 格式 功 能 
mysql_ queryO resource mysql query ( string $query [. resource | 发 送 一 条 MySQL 查询 到 MySQL 
Slink identifier] ) 数据 库 
Imysql db queryO resource mysql db query ( string $database，string | 发 送 一 条 MySQL 查询 到 MySQL 
$query [, resource $ link identifier ] ) 数据 库 
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1. mysql_query() 

mysql queryO 函 数 表示 向 指定 的 连接 标识 符 关 联 的 服务 器 中 的 当前 活动 数据 库 发 送 一 条 查询 语 
句 。 第 一 个 参数 表示 要 执行 的 SQL 语句 ， 第 二 个 参数 表示 资源 标识 符 ， 可 以 用 来 存储 该 函数 返回 的 
资源 。 如 果 没 有 指定 link_identifier 连接 标识 符 ， 则 使 用 上 一 个 打开 的 连接 。 如 果 没 有 打开 的 连接 ， 
该 函数 会 尝试 无 参数 调用 mysql connectO 函 数 ， 建 立 一 个 连接 并 使 用 之 。 查 询 结果 会 被 缓存 。 

mysql queryO 函 数 仅 对 SELECT、SHOW、EXPLAIN 或 DESCRIBE 语句 返回 一 个 资源 标识 符 ， 
如 果 查 询 执行 不 正确 则 返回 False。 对 于 其 他 类 型 的 SQL 语句 ， 如 插入 语句 ，mysql_query0 函 数 会 
在 执行 成 功 时 返回 Tue， 出 错时 返回 False。 非 False 的 返回 值 意味 着 查询 是 合法 的 并 能 够 被 服务 器 
执行 。 这 并 不 说 明 任何 有 关 影 响 到 的 行 数 或 返回 的 行 数 。 很 有 可 能 是 一 条 查询 执行 成 功 了 ， 但 并 未 
影响 到 或 并 未 返回 任何 行 。 该 函数 的 使 用 示例 如 下 所 示 : 


在 上 述 代码 中 创建 了 一 个 查询 语句 “select * from cp ”， 作 为 函数 mysql_query0 的 参数 。 如 果 该 
函数 执行 成 功 ， 会 返回 一 个 记录 集 对 象 ， 其 类 型 为 资源 标识 符 。 
mysql_queryO 函 数 不 仅 可 以 执行 查询 操作 ， 还 可 以 完成 一 些 更 新 操作 ， 如 下 所 示 : 


Sexec="insert into cp values (3, "茶杯 ",5.78,， "中 国 制造 ') "7 
Sresult=mysql query (Sexec) 7 

echo $result; 

mysql close(); 


在 上 述 代码 中 mysql_query0 函 数 执行 了 一 条 插入 语句 ， 如 果 该 语句 执行 成 功 会 返回 True 或 1。 
同样 可 以 利用 该 函数 完成 数据 的 修改 和 删除 。 

2. mysql_db_query() 

Imysql db _ queryO 函 数 根据 查询 结果 返回 一 个 正 的 MySQL 结果 资源 号 ， 出 错时 返回 False。 该 
函数 会 对 INSERT、UPDATE、DELETE 查询 返回 True/False 来 指示 成 功 或 失败 。 参 数 database 表示 
进行 操作 的 数据 库 对 象 ，query 表示 要 执行 的 SQL 语句 ， 最 后 一 个 参数 是 资源 标识 符 。 可 以 这 样 认 
为 ，mysql db _ query0O 函 数 是 mysql_select db0 和 mysql_query0O 的 组 合 ， 但 不 推荐 使 用 该 函数 。 

该 函数 的 功能 和 函数 mysql_query0 相 同 ， 需 要 注意 的 是 此 函数 不 会 切换 回 先前 连接 到 的 数据 
库 。 换 名 话说， 不 能 用 此 函数 临时 在 另 一 个 数据 库 上 执行 SQL 查询 ， 只 能 手动 切换 回来 ， 即 使 用 
SQL 语句 。 该 函数 的 使 用 示例 如 下 所 示 : 


<?php 
$link=mysql connect ("localhost", "root", "root"); 
if(!$1link) 
{ 
echo "数据 库 连 接 失败 "; 
} 
eise 


echo "数据 库 连 接 成 功 "; 
Sexec="insert into cp values (4, ' 茶 杯 ', 5.78, ' 中 国 制 造 ')"; 
$result=mysql db query("com", $exec); 
echo $result; 
mysql close(); 
?> 


在 上 述 代码 中 ， 语 句 “S$result=mysql_db_query("com",$exec)” 表 示 执 行 一 条 插入 语句 ， 其 操作 
的 数据 库 对 象 为 com。 同 样 也 可 以 使 用 该 函数 完成 插入 、 修 改 等 语句 。 


11.3.2 ”获取 和 显示 数据 


在 上 一 小 节 中 , 介绍 了 可 以 向 数据 库 发 送 SQL 查询 语句 , 并 返回 一 个 资源 标识 符 对 象 。 实际 上 ， 
还 可 以 利用 这 个 资源 标识 符 对 象 结 合 相应 的 函数 显示 查询 数据 。 这 些 显示 数据 的 函数 分 别 为 
mysql_result()、mysql] fetch rowO、mysql _fetch_arrayO、mysql_fetch_assocO、mysql_fetch_objectO 。 
其 详细 信息 如 表 11-4 所 示 。 
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表 11-4 ”显示 数据 函数 


名 称 语法 格式 功 能 
mysql resultO mixed mysql_result ( resource $result, int $row [| 取得 结果 数据 
mixed $field] ) 
mysql fetch rowO array mysql fetch row (resource $result ) 从 结果 集中 取得 一 行 作为 枚 举 数组 
mysql fetch array() array mysql fetch_array ( resource $result [. int | 从 结果 集中 取得 一 行 作为 关联 数 
$ result type ] ) 组 ， 或 数字 数组 ， 或 二 者 兼 有 
mysql fetch assocO array mysql fetch assoc (resource $result ) 从 结果 集中 取得 一 行 作为 关联 数组 
mysql fetch objectO object mysql fetch object (resource $result ) 从 结果 集中 取得 一 行 作为 对 象 


1. mysql_result() 

Imysql resultO 函 数 返 回 MySQL 结果 集中 一 个 单元 的 内 容 。 换 句 话说 ， 该 函数 可 以 从 资源 对 象 
result 指定 的 row 中 获取 一 个 指定 的 field 数据 。 其 中 ，row 字段 参数 可 以 是 字段 的 偏 移 量 ，field 表 
示 要 获取 的 字段 名 ， 或 者 是 字段 表 . 字 段 名 (tablename.fieldname)。 如 果 给 列 起 了 别名 (select foo as 
bar from..….)， 则 用 别名 替代 列 名 。 

下 面 创建 一 个 案例 ， 演 示 一 下 显示 数据 库 中 表 的 数据 。 打 开 记 事 本 ， 输 入 下 列 代码 : 

案例 11-2 

<?php 

$link=mysql_ connect ("localhost", "root", "root"); 
if(!$link) 
{ cho "数据 库 连接 失败 "; } 


else 
{ echo "数据 库 连 接 成 功 ";  } 

mysql select db("com"); 
$exec="select * from cp"; 
$result=mysql query ($exec); 
$rs=mysql_ result ($result,2,cp name); 
echo "<br>".$rs; 
mysql_ close(); 

?> 


将 上 述 文件 保存 , 文件 名 为 rsl.php。 打开 正 浏览 器 , 在 地 址 栏 中 输入 http://localhost:8080/Data/ 
rsl.php， 单 击 【 转 到 】 按 钮 ， 会 显示 如 图 11-2 所 示 的 窗口 。 

在 上 述 代 码 中 , 语句 “$rs=mysql result($result,2.cp name)” 
表示 获取 资源 对 象 result 中 的 第 2 行 记录 的 字段 cp_name 的 值 ， 
并 将 返回 的 结果 返回 给 rs。 

使 用 mysql_result0 函 数 不 仅 可 以 显示 单个 字段 的 数值 , 同 


样 也 可 以 显示 整个 表 中 字段 的 数值 。 下 面 创建 一 个 案例 ， 演 示 图 11-2 显示 记录 
一 下 显示 表 中 所 有 的 数据 。 打 开 记 事 本 ， 输 入 下 列 代码 : 
案例 11-3 
<?php 
$link=mysql connect ("localhost", "root", "root"); 
if(!$1link) 


{ echo "数据 库 连 接 失败 "; } 
else { echo "数据 库 连接 成 功 "; } 
mysql select db("com"); 
Sexec="select * from cp"7 
$result=mysql query ($exec); 
for($count=0;$count<mysql numrows ($result);$count++){ 
$id=mysql result ($result, $count, "cp id"); 
$name=mysql result ($result, $count,"cp name"); 
$price=mysql result ($result, $count,"cp price"); 
$des=mysql result ($result, $count, "cp des"); 
echo "<br> 产 品 号 ".$id." 产品 名 称 ".Sname." 产品 价格 ".Sprice." 产品 信息 " .$des; 
} 
mysql close(); 
> 


将 上 述 代码 保存 , 文件 名 为 rs2.php。 打开 正 浏 览 器 , 在 地 址 栏 中 输入 http://localhost:8080/Data/ 
rs2.php， 单 击 【 转 到 】 按 钮 ， 会 显示 如 图 11-3 所 
示 的 窗口 。 

在 上 述 代码 中 ， 使 用 了 一 个 for 循环 把 表 中 
的 数据 全 部 显示 了 出 来 ， 其 中 语句 “mysql_ 
numrows($resulb ”表示 获取 该 资源 对 象 中 记录 的 
数目 。 函 数 mysql numrowsO 可 以 获取 SELECT 
语句 返回 的 记录 数 ， 即 该 查询 语句 的 执行 共有 多 图 11-3 显示 全 部 数据 
少 个 记录 满足 条 件 。 语 句 “$id=mysql_result($result,$count,"cp_id")” 表 示 获 取 每 行 指定 字段 的 值 。 

在 本 案例 中 显示 了 表 中 的 全 部 数据 ， 但 每 个 字段 都 要 有 相应 的 语句 获取 。 如 果 表 中 有 30 个 字 
段 或 者 更 多 ， 采 用 这 种 方式 显然 不 可 以 。 幸 运 的 是 ，PHP 还 提供 了 其 他 的 获取 数据 的 函数 。 


\ 
0 [ 如 果 查 询 中 字段 名 是 别名 ， 在 mmysqL_ resultO 函 数 中 就 使 用 别名 ， 而 不 是 实际 的 字段 名 。 

2. mysql_fetch_row() 

Imysql_fetch rowO 函 数 从 指定 的 结果 标识 符 关 联 的 结果 集中 取得 一 行 数据 ， 并 作为 数组 返回 。 
每 个 结果 的 列 存储 在 一 个 数组 的 单元 中 ， 偏 移 量 从 0 开始 。 该 函数 的 返回 值 是 根据 所 取得 的 行 生 成 
的 数组 , 如 果 下 面 没有 记录 则 返回 False。 依次 调用 mysql_fetch rowO 函 数 将 返回 结果 集中 的 下 一 行 ， 
如 果 下 面 没有 记录 则 返回 False。 该 函数 的 使 用 示例 如 下 所 示 : 


<?php 
$link=mysql connect ("localhost","root","root"); 
if(!$1link) 
{ echo "数据 库 连 接 失败 "; } 
else 


{ echo "数据 库 连接 成 功 ";  } 
mysql select db("com"); 
$exec="select * from cp"; 
$result=mysql] query ($exec); 
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while(list(S$Sid, $name, $price, $des)=mysql fetch row($result)){ 
echo "<br> 产 品 号 " .$id."” 产品 名 称 ".Sname." 产品 价格 ".Sprice." 产品 信息 " 
.$des; 
} 
mysql close(); 
eS 


在 上 述 代码 中 ， 使 用 list0 函 数 获取 函数 mysql_fetch rowO 从 表 中 读 取 的 每 行 数据 ， 并 输出 listO 
函数 的 值 。mysql fetch_row0 函 数 每 次 读 取 一 行 ， 会 自动 判断 并 向 下 移动 。 如 果 下 面 没有 该 记录 ， 
则 返回 False。 如 果 存 在 记录 ， 则 继续 向 下 移动 。 

3. mysql_fetch_array() 

该 函数 的 返回 值 是 根据 从 结果 集中 取得 的 行 生 成 的 数组 ， 如 果 资 源 对 象 中 没有 记录 则 返回 
False。 第 一 个 参数 表示 获取 数据 的 资源 对 象 ， 第 二 个 参数 表示 是 可 选 的，result_ type 是 一 个 常量 ， 
可 以 接受 以 下 值 : MYSQL ASSOC、MYSQL NUM 和 MYSQL BOTH, 默认 值 是 MYSQL _ BOTH。 
如 果 使 用 MYSQL BOTH， 将 得 到 一 个 同时 包含 关联 和 数字 索引 的 数组 。 用 MYSQL ASSOC 只 得 
到 关联 索引 如同 mysql_fetch_assocO 函 数 那 样 )， 用 MYSQL NUM 只 得 到 数字 索引 如同 
mysql_fetch_rowO 函 数 那样 )。 

mysql_fetch_array0 函 数 是 mysql_fetch_row0 函 数 的 扩展 版 本 ， 除 了 将 数据 以 数字 索引 方式 存储 
在 数组 中 之 外 ， 还 可 以 将 数据 作为 关联 索引 存储 ， 用 字段 名 作为 键 名 。 如 果 结 果 集 的 两 个 或 两 个 以 
上 的 列 具有 相同 字段 名 ， 最 后 一 列 将 优先 。 要 访问 同名 的 其 他 列 ， 必 须 用 该 列 的 数字 索引 或 给 该 列 
起 个 别名 。 对 有 别名 的 列 ， 不 能 再 用 原来 的 列 名 访问 其 内 容 (如 本 例 中 的 field)。 

mysql_fetch_array0 函 数 的 使 用 示例 如 下 所 示 : 
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<?php 
$link=mysql connect ("localhost", "root", "root"); 
if(!$1link) 
{ echo "数据 库 连 接 失败 "; } 
else 


{ echo "数据 库 连 接 成 功 ";  } 

mysql select db("com"); 

Sexec="select * from cp"; 

$result=mysql query ($exec); 

while ($row=mysql fetch array ($result,MYSQL ASSOC)){ 
$id=$row['cp id']; 
$name=$row['cp_name']; 
$price=$row['cp price']; 
$des=$row['cp des']; 
echo "<br> 产 品 号 " .$id.” 产品 名 称 ". $name .” 产品 价格 ".Sprice." 产品 信息 " 
.$des; 

} 
mysql close(); 
ee 


在 上 述 代码 中 , 使 用 数组 row 获取 函数 mysql_fetch_array0 从 数据 库 表 中 的 数据 , 并 把 获取 的 数 
据 使 用 列 名 输出 来 ， 如 语句 “$id-$row['cp_id]”。 上 述 代码 中 的 while 循环 语句 可 以 用 下 面 的 代码 来 


代替 : 


while (S$row=mysql fetch array(S$result,MYSQL NUM) ) { 
$id=$row[0]; 
Sname=Srow[1]7 
Sprice=Srow[2]7 
$des=$row[3]; 
echo "<br> 产 品 号 " .$id."” 产品 名 称 ".Sname." 产品 价格 ".Sprice." 产品 信息 " 
.$des; 
1 
4. mysql_fetch_assoc() 
Imysql_fetch_assocO 函 数 的 返回 值 是 根据 从 结果 集中 取得 的 行 生 成 的 关联 数组 , 如 果 结 果 集 中 没 
有 下 一 行 则 返回 False。mysql_fetch_assoc0 函 数 和 用 mysql fetch_array0 函 数 加 上 第 二 个 可 选 参数 
MYSQL ASSOC 完全 相同 ， 它 仅仅 返回 关联 数组 。 该 函数 的 使 用 示例 如 下 所 示 : 


<?php 
$link=mysql_ connect ("localhost", "root", "root"); 
if(!$link) 
{ echo "数据 库 连 接 失败 "; } 
else 


{ echo "数据 库 连 接 成 功 ";  } 

mysql select db("com"); 

Sexec="select * from cp"; 

$result=mysql query ($exec); 

While ($row=mysql fetch assoc($result)){ 
$id=$row['cp_id']; 
$name=$row['cp_name']; 
$price=$row['cp price']; 
$des=$row['cp des']; 
echo "<br> 产 品 号 " .$id."” 产品 名 称 ".Sname."” 产品 价格 ".Sprice." 产品 信息 " 
.$des; 

1 
mysql_close()7 
?> 


5. mysql_fetch_object() 

mysql_fetch_objectQ 函 数 的 返回 值 是 根据 所 取得 的 行 生成 的 对 象 ， 如 果 没 有 更 多 行 则 返 
False。mysql fetch_object0 函 数 和 mysql_fetch_array0 函 数 类 似 ， 只 有 一 点 区 别 ， 返 回 一 个 对 象 而 不 
是 数组 。 间 接地 也 意味 着 只 能 通过 字段 名 来 访问 数组 ， 而 不 是 偏 移 量 (数字 是 合法 的 属性 名 )。 

下 面 创建 一 个 案例 ， 演 示 使 用 mysql_fetch_object0 函 数 获得 数据 库 记 录 集 。 打 开 记 事 本 ， 输 入 
下 列 代码 : 

案例 11-4 

<?php 

$link=mysql connect ("localhost", "root", "root"); 


if(!$1link) 
{ echo "数据 库 连 接 失败 "; } 
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else 
{ echo "数据 库 连 接 成 功 "; } 
mysql select db("com"); 
$exec="select * from cp"; 
$result=mysql query ($exec); 
while($rs=mysql fetch object ($result)) 
{ 
$id=$rs->cp id; 
$name=$rs->cp name; 
$price=$rs->cp_price; 
$des=$rs->cp_des; 
echo "<br> 产 品 号 " .$id."” 产品 名 称 ".Sname." 产品 价格 ".Sprice." 产品 信息 " .$des; 
} 
mysql close(); 
2 


将 上 述 代 码 保存 , 文件 名 为 rs5.php。 打开 正 浏览 器 , 在 地 址 栏 中 输入 http://localhost:8080/Data/ 
rs6.php， 单 击 【 转 到 】 按 钮 ， 会 显示 如 图 11-4 所 示 
的 窗口 。 


在 上 述 代码 中 ， 语 名 “$rs=mysql_fetch_ object | 闭 什 jap 加 三 产 sa 
($resulb” 表 示 获 取 资 源 对 象 result 中 的 记录 ， 该 返 时 
回 值 是 一 个 对 象 , 如 果 下 面 没有 记录 , 则 返回 False， 
否则 会 一 直 向 下 移动 。 最 后 利用 该 对 象 获取 数值 并 图 11-4 显示 数据 
输出 。 


\ 
0 | 注意 : 上 述 函 数 返 回 的 字段 名 是 区 分 大 小 写 的 。 
示 


11.3.3 插入 数据 


在 客户 端 注册 会 员 信息 或 者 发 布 评论 ， 都 需要 向 服务 器 提交 数据 。 数 据 的 提交 是 通过 在 数据 库 
表 中 插入 数据 来 实现 的 。 
下 面 创建 一 个 案例 ， 演 示 在 客户 端 提交 数据 并 放 入 数据 库 表 中 。 该 案例 主 
打开 记事 本 ,创建 一 个 提交 页 面 ， 输 入 下 列 代码 : 


案例 11-5 

<html> 

<body> 

<penter > 

<table border=1> 

<caption> 产 品 输入 表 </caption> 

<form action=inse.php method=post> 
<tr><td> 产 品 号 :</td><td><input type=text name=id></td> 
<tr><td> 产 品名 称 : </td><td><input type=text name=name></td> 
<tr><td> 产 品 价格 :</td><td><input type=text name=price></td> 


实现 添加 产品 的 操 
作 


处 理 ins.php 页 面 提交 的 数据 。 在 PHP 中 ， 以 数组 的 形 
式 获取 一 个 页 面 提 交 的 数据 , 如 语句 “$id=$_ POST['id]” 
表示 获取 ins.php 页 面 表单 中 表单 元 素 名 称 为 id 的 值 ， 

POST 必须 大 写 ，$ 和 POST 之 间 存 在 一 个 下 划 线 。 其 他 
各 表单 元 素 的 获取 以 此 类 推 。 当 获取 客户 端 提交 的 数据 
后 ， 
mysql_query0 函 数 完成 数据 的 添加 操作 。 


<tr><td> 产 品 说 明 :</td><td><textarea cols=20 rows=5 name=des></textarea></td> 
<tr><td><input type=submit value= 提 交 ></td><td><input type=reset value= 重 置 > 
</td></tr> 

</form> 

</table> 

</center> 9 

</body> 

</html> 


将 上 述 文件 保存 ， 文 件 名 为 ins.php。 该 文件 主要 实现 了 一 个 产品 的 添加 页 面 。 这 里 需要 注意 的 
要 给 每 一 个 表单 元 素 起 一 个 见 文 知 义 的 名 称 ， 该 表单 中 数据 提交 的 页 面 为 inse.php。 打 开 记 事 
创建 另外 一 个 处 理 页 面 ， 输 入 下 列 代码 : 


案例 11-5 
<?php 
$link=mysql connect ("localhost","root","root"); 
mysql select db("com"); 
$id=$ POST['id']; 
$name=$ POST['name']7 
$price=$_POST[price]; 
$des=$_POsST[des]; 
$exec="insert into cp values ($id,'S$name', $price,'S$des')"; 
$result=mysql query ($exec); 
if($result){ 
echo "产品 已 被 添加 到 表 中 "; 
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y 
else{ 
echo "该 产品 没有 被 添加 ， 请 重新 输入 "; 
} 
mysql close(); 
人 人 
<a href=ins.php> 返 回 添加 页 面 </php> 
> 


将 上 述 代码 保存 ， 文 件 名 为 inse.php。 该 文件 主要 


就 可 以 把 这 些 数据 放 入 SQL 语句 中 ， 并 执行 


打开 正 浏览 器 ， 在 地 址 栏 中 输入 http://localhost: 


ET TREEZTIETTIEII 


8080/Data/ins.php， 单 击 【 转 到 】 按 钮 , 会 显示 如 图 11-5 | 党 全 wnoearer sam 
所 示 的 窗口 。 

在 上 面 的 窗口 中 ， 单 击 【 提 交 】 按 钮 ， 会 显示 如 图 
11-6 所 示 的 窗口 。 图 11-6 提交 成 功 


11.3.4 删除 数据 


删除 数据 库 中 不 需要 或 者 过 期 的 数据 记录 ， 是 数据 库 中 经 常 执行 的 操作 。 本 节 将 以 案例 的 方式 
介绍 在 PHP 页 面 中 实现 数据 库 记 录 的 删除 。 

下 面 创建 一 个 案例 ， 演 示 删 除数 据 库 记录 。 打 开 记 事 本 ， 首 先 创建 显示 数据 库 记 录 的 页 面 ， 输 
入 下 列 代码 : 
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将 上 述 文件 保存 ， 文 件 名 为 dephp。 该 文件 主要 实现 显示 数据 库 中 的 记录 ， 并 提交 删除 的 可 选 
记录 项 。 在 上 述 代码 中 ， 使 用 mysql_fetch_object0 函 数 获取 数据 库 记 录 ， 并 把 表 中 的 数据 保存 到 相 
应 的 变量 中 ， 如 产品 的 名 称 保存 在 变量 name 中 。 在 HIML 代码 中 ， 语 句 “<td><input type=radio 
name=de value=-<?php echo $id?>></td>” 表 示 创 建 一 个 单 选 按钮 ， 作 为 删除 选择 的 选项 ， 该 表单 元 
素 的 值 为 产品 号 。 代 码 “<?php echo $name?>” 表 示 该 区 表格 显示 的 数据 为 产品 名 称 ， 其 他 选项 以 
此 类 推 。 

下 面 创 建 执行 删除 操作 的 PHP 页 面 ， 打 开 记 事 本 ， 输 入 下 列 代码 : 


$link=mysql connect ("localhost","root","root"); 
mysql select db("com"); 
$id=$ POsT['de']; 
Sexec="delete from cp where cp id=$id"; 
$result=mysql query ($exec); 
if((mysql affected rows()==0) || (mysql affected rows()==-1)) . 
和 
echo "没有 找到 记录 ， 或 者 删除 时 产生 错误 "7 
exit; 
else{ 
echo "产品 已 被 删除 "7 
} 
mysql close(); 


二 溃 


?> 


<a href=de.php> 返 回 删除 页 面 </a> 


将 上 述 代码 保存 ， 文 件 名 为 delphp。 在 该 文件 中 ， 
使 用 语句 *$id=$_POST[de]” 获 取 要 删除 选项 的 产品 序 “| ES = 
列 号 , 使 用 mysql_query0 函 数 执行 数据 库 选项 的 删除 操 产品 -上 


选项 证 二 产品 名 称 产品 价格 “产品 说 明 


作 。 为 了 判断 mysql_query0 是 否 执 行 成 功 , 这 里 使 用 了 回 情怀 全 基山 和 
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函数 mysql_affected_ rowsO。 如 果 表 中 没有 符合 操作 条 3 ms 4 和 

件 的 记录 ，mysql_affected_rows() 函 数 返 回 的 结果 为 0， © Lc 译 主 ”23.60 ”让 国 制 兴 

如 果 执行 SQL 语句 时 失败 ， 则 返回 值 为 -1。 2 
打开 正 浏览 器 ， 在 地 址 栏 中 输入 http://localhost: 

8080/Data/de php， 单 击 【 转 到 】 按 钮 ， 会 显示 如 图 11-7 ET 

所 示 的 窗口 。 


在 窗口 中 选中 要 删除 的 选项 ， 如 图 11-7 所 示 ， 然 
后 单 击 【删除 】 按 钮 ， 会 显示 如 图 11-8 所 示 的 窗口 。 


11.3.5 ”修改 数据 


修改 会 员 的 注册 信息 或 者 修改 购物 网 站 的 价格 信 图 11-8 删除 成 功 
息 ， 都 需要 把 修改 的 信息 提交 给 服务 器 操作 。 本 节 将 以 案例 的 形式 介绍 数据 库 记 录 的 修改 操作 。 

下 面 创建 一 个 案例 ， 演 示 修 改 数据 库 记 录 。 首 先 创建 修改 操作 的 显示 页 面 ， 打 开 记事 本 ， 输 入 
下 列 代码 : 


案例 11-7 

<h1> 修 改 界面 </h1> 

<form action =xi.php method=post> 

请 选择 修改 的 产品 号 : <select name=id> 

<?php 
$link=mysql connect ("localhost", "root", "root"); 
mysql select db("com"); 
S$exec="select * from cp"; 
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将 上 述 代码 保存 ， 文 件 名 为 xiu.php。 该 文件 主要 显示 数据 库 表 中 的 产品 号 。 这 里 注意 嵌入 PHP 
脚本 代码 的 位 置 。 
下 面 创建 实现 修改 操作 的 修改 页 面 ， 打 开 记事 本 ， 输 入 下 列 代码 : 


将 上 述 文件 保存 ， 文 件 名 为 xiphp。 该 文件 主要 显示 要 修改 的 数据 记录 项 的 所 有 数据 ， 其 显示 
位 置 在 指定 的 文本 域 中 。 在 上 述 代码 中 ， 使 用 语句 “$resultrmysql query($exec) ”获取 数据 库 表 中 
的 信息 ， 语 句 “$rs=mysql fetch_ object($resulb ”获取 指定 资源 的 信息 ， 并 使 用 对 象 rs 显示 记录 的 信 
息 ， 如 “$des=$rs->cp_des”。 最 后 把 获取 的 产品 信息 作为 数据 放 入 表单 元 素 中 。 

最 后 创建 实现 修改 操作 的 执行 页 面 ， 打 开 记 事 本 ， 输 入 下 列 代码 : 


案例 11-7 
<?php 
$link=mysql_connect ("localhost", "root", "root"); 
mysql select db("com"); 
$id=$ POST['id']7 
$name=$ POST['name']; 
S$price=$ POST[price]; 
$des=$_ POsT[des]; 
$exec="update cp set cp_name='$name',cp price=$price,cp des="'$des' where cp_id 
=$id"; 
$result=mysql query ($exec); 
if($result){ 
echo "产品 已 被 修改 "; 


} 
else{ 
echo "该 产品 没有 被 修 改 "; 
. 
mysql close(); 

学 > 
<hr> 
<a href=ins .php> 返 回 修改 页 面 </php> 


将 上 述 文件 保存 ， 文 件 名 为 x.php。 该 文件 主要 获取 修改 后 的 数据 ， 并 更 新 数据 库 记 录 。 该 页 
面 的 代码 和 实现 插入 页 面 的 代码 基本 一 样 ， 这 里 就 不 再 介绍 了 。 

打开 正 浏览 器 ， 在 地 址 栏 中 输入 http://localhost:8080/Data/xiu.php， 单 击 【 转 到 】 按 钮 ， 会 显示 
如 图 11-9 所 示 的 窗口 。 在 该 窗口 中 选择 要 修改 的 产品 号 , 然后 单 击 【 提 交 】 按 钮 , 会 显示 如 图 11-10 
所 示 的 窗口 。 

在 图 11-10 的 窗口 中 ， 修 改 产品 的 数据 ， 这 里 只 允许 修改 产品 的 名 称 、 价 格 和 说 明 。 当 修改 完 
成 之 后 ， 单 击 【提交 】 按 钮 ， 会 显示 如 图 11-11 所 示 的 窗口 。 


产品 己 被 你 改 


百 


请 选择 修改 的 产品 号 :2 司 
EE) 


图 11-9 选择 修改 项 图 11-10 修改 数据 图 11-11 修改 成 功 


全 部 
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11.4 ”数据 库 高 级 操作 


在 前 面 的 章节 中 介绍 了 PHP 中 数据 库 的 基础 性 操作 ， 如 查询 、 显 示 、 更 新 数据 等 。 对 于 数据 库 
的 操作 远 远 不 止 这 些 ， 还 可 以 在 PHP 中 获取 MySQL 数据 库 的 表 信 息 和 字段 信息 等 。 本 节 将 详细 地 
介绍 PHP 中 数据 库 的 高 级 操作 。 


11.4.1 获取 报错 消息 


创建 一 个 没有 bug 的 程序 ， 是 所 有 程序 员 的 梦想 ， 但 是 不 存在 bug 的 程序 ， 在 现实 中 是 不 存在 
的 ， 除 非 这 个 程序 是 一 段 没有 意义 的 代码 。 获 取 详 细 的 报错 消息 ， 并 根据 错误 原因 ， 解 决 错误 是 必 
须要 做 的 工作 。 在 PHP 5.0 中 ， 提 供 了 两 个 函数 ， 可 以 获取 MySQL 函数 执行 时 可 能 会 发 生 错误 的 
代码 信息 ， 它 们 分 别 为 mysql_errorO0 函 数 和 mysql_ermo0 函 数 ， 其 详细 信息 如 表 11-5 所 示 。 
表 11-5 ”获取 报错 消息 函数 


名 称 功 能 


TIT 
本 报错 消息 
消息 的 数字 编码 


1. mysql_error() 
mysql_error0 函 数 返回 上 一 个 MySQL 函数 的 错误 文本 ， 如 果 没 有 出 错 则 返回 "( 空 字符 串 )。 如 
果 没 有 指定 连接 资源 号 ， 则 使 用 上 一 个 成 功 打开 的 连接 从 MySQL 服务 器 提取 报错 消息 。 从 MySQL 
数据 库 后 端 来 的 错误 不 再 发 出 警告 ， 要 用 mysql_errorO 函 数 来 提取 错误 文本 。 注 意 本 函数 仅 返回 最 
近 一 次 MySQL 函数 的 执行 〈 不 包括 mysql_error0 函 数 和 mysql_ermo0 函 数 ) 的 错误 文本 ， 因 此 如 
果 要 使 用 此 函数 ， 确 保 在 调用 另 一 个 MySQL 函数 之 前 检查 它 的 值 。 该 函数 的 使 用 示例 如 下 所 示 : 
<?php 
mysql_connect ("localhost", “root", "root"); 
mysql select db ("nonexistentdb") 7 
echo mysql error(). "<br>"; 
mysql select db("com"); 
mysql query("SELECT * FROM nonexistenttable"); 
echo mysql error() . "<br>"; 
[人 


在 上 述 代码 中 ， 如 果 在 MySQL 服务 器 中 选择 一 个 不 存在 的 数据 库 ， 就 会 产生 一 个 报错 消息 ， 
其 报错 消息 是 mysql_errorO 函 数 产 生 的 。 同 样 ， 在 已 有 的 数据 库 中 ， 对 一 个 不 存在 的 数据 库 表 操作 
也 会 产生 错误 。 

该 段 代码 执行 后 的 结果 是 在 PHP 页 面 上 ， 输 出 错误 产生 的 文本 信息 ， 其 信息 如 下 所 示 : 


Unknown database 'nonexistentdb"' 
Table 'com.nonexistenttable' doesn't exist 


\ 
[ MySQL 的 报错 消息 有 20 种 语言 ， 存 储 在 MYSQL _ INSTALL DIR/mysq/LANGUAGE. 
提示 


弹 


2. mysql_errno() 
mysql_ermo0 函 数 返 回 上 一 个 MySQL 函数 的 错误 号 码 , 如 果 没 有 出 错 则 返回 0( 零 )。 从 MySQL 
数据 库 后 端 来 的 错误 不 再 发 出 警告 ， 要 用 mysql_errmo0 函 数 来 提取 错误 代码 。 注 意 该 函数 仅 返 回 最 
近 一 次 MySQL 函数 的 执行 〈 不 包括 mysql_error0 函 数 和 mysql ermo0 函 数 ) 的 错误 代码 ， 因 此 如 
果 要 使 用 此 函数 ， 确 保 在 调用 另 一 个 MySQL 函数 之 前 检查 它 的 值 。 该 函数 的 使 用 示例 如 下 所 示 : 
<?php 
mysql connect ("localhost", "root", "root"); 


mysql select db("com"); 
mysql query ("SELECT * FROM cp where aa=3"); 


echo msql erraot  "S 二 YSGI error(} “Nn 
re 
在 上 述 代码 中 ， 当 在 指定 的 数据 库 com 中 ， 查 询 cp 表 中 一 个 不 存在 的 字段 aa 时 ， 就 会 产生 一 
个 报错 消息 。 可 以 使 用 mysql_ermo0 函 数 将 产生 错误 的 号 码 输出 来 。 该 段 代 码 的 执行 结果 为 “1054: 


1 


Unknown column 'aa' in "where clause'”。 


11.4.2 ”获取 数据 库 和 表 信 息 


在 PHP 中 ， 获 取 MySQL 服务 器 中 数据 库 和 表 的 信息 ， 通 常 使 用 下 列 4 个 函数 ， 它 们 分 别 为 
mysql_list_dbs()、mysql db_name0、mysql_list_ tables0 和 mysql_table name0。 其 详细 信息 如 表 11-6 
所 示 。 

表 11-6 ”获取 数据 库 系统 信息 函数 


名 称 功 能 
mysql list_dbsO 列 出 服务 器 中 所 有 的 数据 库 
mysql_ db nameO) string mysql db name ( resource $result, int $row [. | 取得 结果 数据 
mixed $field] ) 
mysql_list_tables0) resource mysql list_ tables ( string $database [. resource | 列 出 数据 库 中 的 表 


Slink identifier] ) 
mysql_table name string mysql_tablename ( resource $result. int $i ) 取得 表 名 


1. mysql_list_dbs() 
mysql_list_dbs0 函 数 将 返回 一 个 结果 指针 ， 包 含 了 当前 MySQL 进程 中 所 有 可 用 的 数据 库 。 该 
函数 参数 为 资源 标识 符 。 该 函数 的 使 用 示例 如 下 所 示 : 


<?php 
$link = mysql connect ("localhost", "root", "root"); 
$db list = mysql list dbs ($link); 
while ($row = mysql fetch object($db list)) { 
echo $row->Database . "<br>"; 


1 

2 

在 上 述 代码 中 创建 了 指向 服务 器 的 资源 标识 符 对 象 ink， 以 link 作为 参数 ，mysql list dbsO 函 

数 获取 一 个 指针 对 象 ， 并 使 用 while 循环 输出 该 服务 器 中 存在 的 数据 库 。 

2. mysql_db_name() 

mysql_db_name0 〇 函数 用 于 取得 mysql_list dbsO 函 数 调用 所 返回 的 数据 库 名 。 参 数 result 表示 
中 Imysql_list_ dbsO 函 数 调用 所 返回 的 结果 指针 。 人 参数 row 表示 结果 集中 的 行 号 。 参 数 field 表示 结果 集 
中 的 字段 名 。 该 函数 如 果 执 行 成 功 则 返回 数据 库 名 ， 和 否则 返回 False。 如 果 返 回 了 False， 则 可 以 使 
用 mysql_errorO 函 数 来 判断 错误 的 种 类 。 该 函数 的 使 用 示例 如 下 所 示 : 


这 <?php 

Eo $link = mysql_ connect ("localhost", "root", "root"); 
Es $db list = mysql list dbs ($1link); 

[92] $i= 0; 

如 $cnt = mysql num rows($db list); 

完 while ($i < $cnt) { 

全 echo mysql db name($db list, $i) . "\n"; 

学 $itty 

名 上 


在 上 述 代码 中 ，mysql_ num rows0 函 数 表示 获取 该 结果 集中 的 行 数 。 

3. mysql_list_tables() 

mysql list_tablesO 函 数 接收 一 个 数据 库 名 并 返回 和 mysql_query0 函 数 很 相似 的 一 个 结果 指针 。 
用 mysql_tablename0) 函 数 来 遍历 此 结果 指针 , 或 者 任何 使 用 结果 表 的 函数 , 例如 mysql_fetch_array0O。 
database 参数 是 需要 被 获取 表 名 的 数据 库 名 。 该 函数 如 果 执 行 失败 则 返回 False。 该 函数 的 使 用 示例 
如 下 所 示 : 


<?php 
$link = mysql_connect ("localhost", "root", "root"); 
$result = mysql list tables (Com) 7 
if (!Sresult) { 
print "DB Error, could not list tables\n"; 
print "MYSQL Error: ' . mysql error(); 
exit; 
} 
while ($row = mysql fetch row(Sresult)) { 
print "Table: $row[0]\n"; 
} 
mysql free result ($result); 
2 


在 上 述 代 码 中 ， 函 数 mysql free _ result0 表 示 释 放 内 存 。 
4. mysql_tablename 
Imysql tablename() 函 数 接收 mysql list_tablesO 函 数 返 回 的 结果 指针 ， 以 及 一 个 整数 索引 作为 参 
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数 并 返回 表 名 。 该 函数 的 使 用 示例 如 下 所 示 : 


<?php 
mysql connect ("localhost", “root", "root"); 
$result = mysql list tables("com"); 3 
for ($i = 0; $i < mysql num rows (Sresult) 7 $i++) 让 


printf ("Table: $s\n", mysql tablename ($result, $i)); 
mysql free result ($result); 
2 
在 上 述 代 码 中 , 可 以 用 mysql_num rows0 函 数 来 判断 结果 指针 中 的 表 的 数目 。mysql_tablename() 
函数 来 遍历 此 结果 指针 ， 或 者 任何 处 理 结果 表 的 函数 。 


11.4.3 ”获取 字段 信息 


全 小 


在 PHP 中 ， 获 取 MySQL 服务 器 表 中 字段 的 信息 ， 如 字段 的 名 称 、 数 据 类 型 等 ， 主 要 使 用 下 面 
的 函数 ， 分 别 为 mysql_fetch _ field0、mysql_num fields()、mysql list_fieldsO0、mysql field_flagsO)、 
mysql_field_len0、mysql_field_name0、mysql_field_ typeO0 和 mysql_field table0。 其 详细 信息 如 表 11-7 
所 示 。 
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表 11-7 ”获取 字段 信息 函数 


名 称 | 语法 格式 | 功 能 


mysql_fetch field() | object mysql fetch field (resource $result [. int $field_offset] ) ”| 从 结果 集中 取得 列 信息 并 
作为 对 象 返 回 

mysql num fields0 | int mysql num fields ( resource $result ) 取得 结果 集中 字段 的 数目 

mysql list_fields| resource mysql list fields ( string $database name， string | 获取 给 定 表 名 的 信息 

S$table_name [, resource $link identifier] ) 

mysql field flags)0 | string mysql field flags (resource $result, int $field offset ) 从 结果 集中 取得 和 指定 字 
段 关 联 的 标志 

mysql_field_len0 int mysql_field len ( resource Sresult int $field_offset ) 返回 指定 字段 的 长 度 

mysql field name() | string mysql field name (resource $result. int $field index ) 取得 结果 集中 指定 字段 的 
字段 名 

mysql field typeO string mysql field_type ( resource Sresult int $field_offset ) 取得 结果 集中 指定 字段 的 
数据 类 型 

mysql field table) | string mysql field table (resource Sresult int $field offset ) 取得 指定 字段 所 在 的 表 名 


1. mysql_fetch_field() 

mysql fetch field0 函 数 返 回 一 个 包含 字段 信息 的 对 象 。 可 以 用 来 从 某 个 查询 结果 中 取得 字段 的 
信息 。 参 数 field_offset 表示 字段 偏 移 量 ， 如 果 没有 指定 字段 偏 移 量 ， 则 下 一 个 尚未 被 mysql_fetch_ 
field0 函 数 取 得 的 字段 被 提取 。 

mysql fetch_field0 函 数 返 回 的 是 一 个 字段 信息 对 象 ， 返 回 对 象 具 有 下 列 属性 ， 其 详细 信息 如 
表 11-8 所 示 。 
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表 11-8 ”字段 属性 


名 称 含义 
name 列 名 
table 所 在 的 表 名 
max_length 最 大 长 度 
not_null 如 果 该 列 不 能 为 NULL 则 为 1， 否则 为 0 
Primary key 如 果 该 列 是 primary key 则 为 1， 否 则 为 0 
D unique key 如 果 该 列 是 unique key 则 为 1， 否 则 为 0 
multiple key 如 果 该 列 是 non-unique key 则 为 1， 否 则 为 0 
numeric 如 果 该 列 是 numeric 则 为 1， 否 则 为 0 
加 blob 如 果 该 列 是 BLOB 则 为 1， 否则 为 0 
主 type 该 列 的 数据 类 型 
S unsigned 如 果 该 列 是 无 符号 数 则 为 1， 否 则 为 0 
如 zerofill 如 果 该 列 是 zero-filled ( 零 填充 ) 则 为 1， 否则 0 
三 以 上 的 属性 在 使 用 过 程 中 ， 区 分 大 小 写 。 下 面 创建 一 个 案例 ， 演 示 获 取 表 中 的 信息 。 打 开 记 事 
习 本 ， 输 入 下 列 代码 : 
案例 11-8 
<?php 


mysql_connect ("localhost", "root", "root") 
or die("Could not connect: " . mysql error()); 
mysql select db("com"); 
$result = mysql query("select * from cp") 
or die("Query failed: " . mysql error()); 
$i = 07 
while ($i < mysql num fields (Sresult)) { 
echo "该 表 中 的 字段 分 别 为 ， 第 $1: 个 字段 <br/>\n"; 
Smeta = mysql fetch field(Sresult) 
if (!Smeta) { 
echo " 表 中 没有 信息 存在 <br/>\n"; 
} 
echo "<pre> 
字段 名 称 : $meta->name 
字段 数据 类 型 : Smeta->type 
字段 的 长 度 : $meta->max_length 
</pre>"; 
Si+ 二 7 
4 
mysql free result ($result); 
> 


将 上 述 文件 保存 , 文件 名 为 fieldphp。 打 开 正 浏览 器 , 在 地 址 栏 中 输入 http://localhost:8080/Data/ 
field.php， 单 击 【 转 到 】 按 钮 ， 会 显示 如 图 11-12 所 示 的 窗口 。 
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在 本 案例 中 , 使 用 到 了 语句 “or die("Could 
not connect: " . mysql error())”， 该 语句 表示 如 
果 上 一 个 语句 没有 执行 成 功 ， 就 执行 该 语句 并 
输出 产生 的 报错 消息 。 函 数 mysql_num fields 季度 :2 
(Sresulb 主要 用 来 获取 表 中 字段 的 个 数 。 | 下 "m7839%, 逢 :iF 朋 天 
mysql fetch field(Sresulb) 函 数 每 获取 一 个 字段 和 
对 象 Smeta， 就 输出 该 对 象 的 相应 属性 。 

2. mysqLnum_fields() 


mysql_num fieldsO) 函 数 返回 指定 资源 的 11-12 字段 信息 
字段 数目 。 该 函数 的 示例 如 下 所 示 : 第 
<?php 11 


mysql_connect ("localhost", "root", "root"); 
mysql select db("com"); 

$result = mysql query("select * from cp"); 

echo " 表 cp 有 ".mysql_num fields ($result) ." 个 字段 "; 
mysql free result ($result); 

en 
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3. mysql_list_fields() 

mysql_list_fields() 函 数 表示 取得 给 定 表 名 的 信息 。 参 数 是 数据 库 名 和 表 名 。 返 回 一 个 结果 指针 ， 
可 以 用 于 mysql field_flagsO0、mysql_ field len0、mysql_ field name( 和 mysql_field_type0 等 函数 。 
该 函数 的 使 用 示例 如 下 所 示 : 


<hr color=blue> 
<?php 
$link = mysql_ connect ("localhost", "root", "root"); 
$fields = mysql list fields("com", "cp", $link); 
$columns = mysql num fields ($fields); 
for ($i = 0; $i < $columns; $i++) { 

echo mysql field name ($fields, $i) . "<br>"; 


WD 


?> 

4. mysql_field_flags() 

mysql field_flags0 函 数 返回 指定 字段 的 字段 标志 。 每 个 标志 都 用 一 个 单词 表示 ， 之 间 用 一 个 空 
格 分 开 。 如 果 MySQL 版 本 足够 新 ， 则 会 支持 以 下 标志 : not null、primary key、unique key、 
multiple_ key、blob、unsigned、zerofill、binary、enum、auto_increment、timestamp。 该 函数 的 使 用 
示例 如 下 所 示 : 

<?php 

mysql connect ("localhost", "root", "root"); 


mysql select db("com"); 
$result = mysql] query("select + from cp"); 


$row=mysql fetch row($result); 
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5. mysql field_len() 
mysql field lenO 函 数 返回 指定 结果 集中 指定 字段 的 长 度 ， 参 数 field_offset 表示 字段 偏 移 量 。 该 
函数 的 使 用 示例 如 下 所 示 : 


6. mysql_field_name() 
mysql_field_name() 函 数 返 回 指定 字段 索引 的 字段 名 。result 必须 是 一 个 合法 的 结果 标识 符 ， 
field_index 是 该 字段 的 数字 偏 移 量 。 注 意 ，field_index 从 0 开始。 该 函数 的 使 用 示例 如 下 所 示 : 


7. mysql_field_type() 
mysql_field_typeO 函 数 和 mysql_field_nameO 函 数 相 似 。 参 数 完全 相同 ， 但 返回 的 是 字段 类 型 而 
不 是 字段 名 。 字 段 类 型 有 int、real、string、blob 以 及 其 他 。 该 函数 的 使 用 示例 如 下 所 示 : 


8. mysql_field_table() 
Imysql_ field table0 函 数 返 回 指定 result 结果 集中 指定 偏 移 量 的 字段 所 在 表 的 表 名 。 该 函数 的 使 
用 示例 如 下 所 示 : 


echo mysql field table (Sresult,0) 7 
更 学 


11.4.4 ”辅助 函数 


在 PHP 中 ， 还 存在 许多 与 系统 有 关 的 函数 ， 如 提供 客户 端 信息 或 者 MySQL 服务 器 端 信息 的 函 
数 。 其 详细 信息 如 表 11-9 所 示 。 
表 11-9 辅助 函数 


名 称 语法 格式 功 能 


mysql_client_encoding() | string mysql_client_encoding ([resource $link identifier] ) 返回 字符 集 的 名 称 
Inysql_get_server info() | string mysql_get_server info ( [resource $link identifier] ) | 取得 MySQL 服务 器 信息 
mysql_get_host_infoO) | string mysql get_host_info ( [resource $link identifier] ) | 取得 MySQL 主机 信息 


mysql get client info0 | string mysql get client info (void ) 取得 MySQL 客户 端 信息 


mysql_stat() string mysql stat ( [resource $link identifier] ) 取得 当前 系统 状态 


1. mysql_client_encoding() 
mysql_client_encoding0 函 数 从 MySQL 中 取得 character_set 变量 的 值 ， 该 结果 值 是 一 个 字符 集 
的 名 称 。 该 函数 的 使 用 示例 如 下 所 示 : 


<?php 

$link = mysql connect ("localhost", "root", "root"); 
$charset = mysql_ client encoding ($1ink); 

echo "当前 使 用 的 字符 编码 集 为 : $charset<br>"; 


?> 


2. mysql_get_server_info() 
Inysql_get_server info0 函 数 返 回 link_identifier 所 使 用 的 服务 器 版 本 信息 。 该 函数 的 使 用 示例 如 
下 所 示 : 


<?php 
mysql connect ("localhost", "root"， "root") or 
die("Could not connect: " . mysql error()); 
printf ("MySQL server version: %s\n", mysql get server info()); 
区 


3. mysql_get_host_info() 
Inysql_get_host_ info0 函 数 返回 一 个 字符 串 ， 说 明了 连接 link_identifier 所 使 用 的 连接 方式 , 包括 
服务 器 的 主机 名 。 如 果 省 略 link identifier， 则 使 用 上 一 个 打开 的 连接 。 该 函数 的 使 用 示例 如 下 所 示 : 


<?php 
mysql connect ("localhost", "root", "root") or 
die("Could not connect: " . mysql error()); 
printf ("MySQL host info: %s<br>", mysql get host info()); 
及 二 
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4. mysql_get_client_info() 

Iysql_get_client info0 函 数 返回 一 个 字符 串 ， 表 示 客 户 端 库 的 版 本 。 该 函数 的 使 用 示例 如 下 
所 示 : 

<?php 


printf ("MySQL client info: %s\n", mysql get client info()); 
二 


5. mysql_stat() 

mysql_stat0 函 数 返 回 当 前 服务 器 状态 。 注意 ,mysql_stat0 函 数目 前 只 返回 正常 运行 时 间 、 线程、 
执行 的 查询 、 太 慢 的 查询 、 己 打开 的 表 、 己 刷新 的 表 、 当 前 打开 的 表 和 每 秒 平均 查询 数 等 。 要 得 到 
其 他 状态 变量 的 完整 列表 ， 只 能 使 用 SQL 命令 SHOW STATUS。 该 函数 的 使 用 示例 如 下 所 示 : 

<?php 

$link = mysql connect ("localhost", "root", "root"); 

$status = explode(' ', mysql stat ($link)); 


print r($status); 
区 之 


11.5 PHP 的 MySQLI 扩展 


前 面 的 章节 介绍 了 使 用 标准 的 php_mysql.dll 库 ，PHP 可 以 完成 对 数据 库 的 操作 ， 如 查询 、 更 新 
数据 等 。 PHP 还 提供 了 另外 一 个 扩展 类 库 php_mysqli.dll, 也 可 以 完成 对 数据 库 的 操作 。php_mysql.dll 
类 库 是 基于 过 程 化 的 接口 ， 而 php_mysqli.dll 类 库 是 基于 内 置 的 面向 对 象 接口 。 面 向 对 象 的 接口 不 
仅 能 与 其 他 应 用 程序 更 紧密 地 集成 ， 还 能 根据 需要 扩展 此 接口 。 幸 运 的 是 ， 如 果 非 常熟 悉 MySQL 
库 函 数 ， 学 习 和 使 用 MySQLI 库 函 数 就 非常 容易 上 手 ， 因 为 改进 的 MySQLI 扩展 函数 的 命名 约定 和 
原来 的 函数 几乎 相同 ， 例 如 数据 库 的 连接 函数 mysqli connect0 和 MySQL 库 中 函数 mysql_connectO 
相同 。 此 外 ， 类 似 函 数 的 所 有 参数 和 行为 从 外 部 看 也 没有 什么 不 同 。 本 节 不 再 对 每 一 个 函数 进行 介 
绍 ， 而 将 以 案例 的 形式 ， 介 绍 MySQLI 扩展 包 中 函数 的 使 用 。 


11.5.1 ”MySQLI 的 启用 和 使 用 


PHP 开发 人 员 在 原来 的 基础 上 , 修改 了 mysqli 的 扩展 。mysqli 类 库 不 仅 扩展 了 内 部 行为 来 提升 
性 能 ， 还 加 入 了 一 些 额 外 的 功能 ， 来 方便 用 户 使 用 更 新 后 MySQL 版 本 的 特性 。MySQLI 扩展 包 在 
下 面 几 个 方面 做 了 改进 ， 分 别 为 面向 对 象 、 准 备 语句 、 事 务 支持 、 改 进 的 调试 功能 、 主 /从 支持 等 。 
其 中 面向 对 象 就 是 将 MySQLI 扩展 封装 到 一 个 类 中 ， 从 而 鼓励 使 用 面向 对 象 方法 ; 准备 语句 主要 处 
理 重 复 执行 的 SQL 语句 。 

在 Windows 中 ， 启 用 php_mysqli.dll 扩展 非常 简单 ， 和 启用 php_mysqldll 一 样 ， 都 是 在 php.ini 
文件 中 操作 的 ， 找 到 该 库 后 ， 去 掉 前 面 的 “; ”号 就 可 以 了 。 可 以 看 到 在 11.1 节 的 php.ini 的 代码 中 ， 
在 启用 php_mysql.dll 库 时 ， 把 php_mysqli.dll 启用 就 可 以 了 。 

当 MySQLI 扩展 库 被 启用 之 后 ， 就 可 以 直接 使 用 该 库 中 的 函数 了 。MySQLI 扩展 库 中 的 函数 大 


多 和 MySQL 中 的 函数 命名 相似 。 扩 展 包 中 常用 的 函数 和 方法 如 表 11-10 所 示 。 
表 11-10 ”扩展 包 常 用 函数 


名 称 语法 格式 功 能 对 象 格式 。 
mysqli_connect() mysqli ”mysqli_connect([string ”host | 连接 MySQL 服务 器 | class mysqlif : 
[string usename [. string paswd [. string connect([string host oe. 


dbname [. int port [string socket]]]]]]) [string usename [, 
string paswd [. string 


dbname [, int port 


[string socket]]]]ID) 
} 
mysqli_ select db() boolean mysqli_select_db(mysqli link | 选择 一 个 数据 库 class mysqli { 
string dbname) bool select db ( string 第 
$dbname ) 章 
} 
mysqli_closeO boolean mysqli_close(mysql link) 关闭 数据 库 连 接 class mysqli { 总 
bool close (void ) a 
} 机 
数 
mysqli_connect_error() string mysqli_connect_error(mysqli 产生 错误 消息 , 如 果 | 无 
link) 没有 错误 则 产生 一 
个 空 的 字符 串 
mysqli_connect_ermo() int mysqli_connect ermo() 产生 错误 的 所 在 位 | 无 


置 ， 即 行 号 


下 面 创建 一 个 案例 ， 演 示 使 用 MySQLI 库 中 的 函数 连接 的 MySQL 数据 库 。 打 开 记 事 本 ， 输 入 
下 列 代码 : 


案例 11-9 
<?php 
$link=mysqli connect ("localhost","root", "root"); 
if(!$1link) 
echo mysqli_connect_error() ." 所 在 位 置 为 ".mysqli_connect errno(); 
$link->select_db("com") or die(" 找 不 到 该 数据 库 "); 
echo "数据 库 连 接 成 功 "; 
$1link->close (); 
2 


将 上 述 代 码 保存 ， 文 件 名 为 mysqliphp。 打 开 正 浏览 器 ， 在 地 址 栏 中 输入 http://localhost:8080/ 
Data/mysqliphp, 单 击 【 转 到 】 按 钮 , 会 显示 如 图 11-13 


所 示 的 窗口 。 i ET TE 
了 


在 本 案例 中 ， 使 用 函数 mysqli connectO 函 数 创 
建 了 一 个 数据 库 连 接 对 象 link， 该 函数 具有 三 个 参 
数 ， 第 一 个 参数 表示 连接 的 服务 器 的 位 置 ， 后 面 两 个 
参数 表示 MySQL 数据 库 的 用 户 的 名 称 和 密码 。 函 数 图 11-13 连接 数据 库 
mysqli connect_error0 和 mysqli connect ermo0 分 别 可 以 产生 错误 的 文本 信息 和 行 号 。 和 前 面 的 
MySQL 包 中 的 函数 使 用 一 样 。“ $link->select_db("com")” 表 示 选 择 一 个 要 操作 的 数据 库 对 象 ， 
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“9$link->close0” 表 示 关 闭 数据 库 连 接 。 因 为 select db0、close0 方 法 是 对 象 link 所 具有 的 ， 故 该 方 


法 的 调用 方式 是 使 用 对 象 的 调用 方式 。 


从 上 面 的 案例 可 以 看 出 ， 使 用 MySQLI 扩展 包 连 接 数据 库 的 流程 和 方式 ， 与 前 面 的 MySQL 包 
没有 什么 大 的 区 别 ， 只 是 在 对 个 别 方法 的 调用 时 有 些 差 异 。 


11.5.2 ”MySQLI 查询 


使 用 MySQLI 扩展 库 中 的 函数 获取 和 显示 数据 库 表 中 的 信息 ， 和 MySQL 库 一 样 简单 ， 其 流程 
基本 一 致 。 在 使 用 扩展 包 查 询 的 过 程 中 ， 常 用 函数 和 方法 如 表 11-11 所 示 。 


表 11-11 ”获取 数据 函数 
名 称 语法 格式 功 能 对 象 格式 
mysqli queryO mixed mysqli query(mysqli 执行 一 个 SQL 语句 class mysqli{ 
link,string query[int mixed query(string 
resultmode]) query[,int resultmode]) 


mysqli fetch_objectO 


class mysqli_result{ 
array fetch_objectO 


从 结果 集 返 回 一 个 对 象 ， 
而 不 是 数组 


mysqli fetch rowO 


mysqli fetch arrayO 


mixed mysqli fetch_object 
(mysqli_reuslt $result ) 


mixed mysqli_fetch row 从 结果 集中 获取 一 行 数 
(mysqli_result result) 据 , 并 将 该 值 放 到 一 个 索 
引 数组 中 


mixed mysqli_fetch_array 从 结果 集 获 取 记录 , 并 放 
(mysqli_result $result [, int 到 数组 中 


} 

class mysqli_result{ 
mixed fetch row() 

} 

class mysqli_result { 

mixed fetch array ( [int 


Sresulttype] ) 
array mysqli fetch_assoc 


(mysqli_result $result ) 


mysqli_ fetch assoc() 


Sresulttype] )} 
class mysqli_result { 
array fetch_assoc (void ) 


} 


从 结果 集 获取 记录 , 并 放 
到 数组 中 


mysqli_query0 函 数 的 作用 是 执行 一 个 SQL 语 句 , 其 resultmode 参 数 主 要 用 来 修改 此 方法 的 行为 ， 
有 两 个 值 ， 第 一 个 值 为 MYSQLI_STORE_RESULT 表示 该 结果 作为 缓冲 集合 返回 ， 第 二 个 值 为 


MYSQLI_USE_RESULT 表示 作为 非 缓冲 集合 返 


回 


下 面 创建 一 个 案例 ， 演 示 使 用 MySQLI 扩展 获取 并 显示 数据 。 打 开 记 事 本 ， 输 入 下 列 代码 : 


案例 11-10 
<?php 


$1link=mysqli_ connect ("localhost", "root", "root"); 


if(!$1ink) 


echo mysqli connect error(). "所 在 位 置 为 " -mysqli_connect_errno () 7 
$link->select_db("com") or die(" 找 不 到 该 数据 库 ") ; 


Squery="select * from cp"; 
$result=$1ink->query ($query); 


While ($row=mysqli fetch object ($result)){ 


$id=$row->cp_id; 
$name=$row->cp_name; 


Sprice=Srow->cp price; 
$des=$row->cp des; 
echo "” <br> 产 品 号 ".S$id." 产品 名 称 ". $name ."” 产品 价格 ".$price."” 产品 说 明 ".$des; 


} 
$1link->close (); 
yi . 


将 上 述 文件 保存 ， 文 件 名 为 mysqll php。 打 开 正 浏览 器 ， 在 地 址 栏 中 输入 http://localhost:8080/ 
Data/mysqlil.php, 单 击 【 转 到 】 按 钮 , 会 显示 如 图 11-14 
所 示 的 窗口 。 

在 本 案例 中 ,创建 查询 语句 query 作为 参数 ， 语 句 


产品 号 3 产品 名 称 某 怀 产品 价格 5. 73 产品 i 

“SresulfrSlinke>query(Squery)” 表 示 执行 数据 库 的 查询 “| 各。 :全 于 二 第 
操作 ， 并 返回 一 个 记录 集 对 象 result。 这 里 执行 SQL 语 | “生生 和 1 
名 的 ， 不 是 函数 mysql_query0 而 是 对 象 link 中 的 方法 | 
query0 〇 0。 返回 记录 和 集 result 后 ， 使 用 方法 fetch_objectO 图 11.14 显示 数据 和 
把 记录 集 的 数据 以 数组 的 形式 赋 给 row, 然后 依次 输出 雪 
数组 的 对 象 。 数 

while 循环 语句 的 条 件 可 以 用 下 面 的 语句 “while($row=$result->fetch_object0)” 人 代替， 循环 体 中 访 


采用 mysqli_fetch_object0 函 数 ， 该 函数 也 是 MySQLI 扩展 中 的 一 个 ， 其 执行 机 制 是 基于 过 程 的 。 从 
这 里 可 以 看 出 ，MySQLI 扩展 中 有 两 种 类 型 的 函数 ， 一 种 是 基于 过 程 的， 如 mysqli_fetch_objectO， 
一 种 是 基于 对 象 的 ， 如 fetch_objcetO。 

此 处 不 仅 可 以 使 用 query0 方 法 执行 SQL 语句 ， 还 可 以 使 用 mysqli_query0 函 数 。mysqli_query0 
函数 使 用 的 是 过 程 化 的 语法 。 同 样 ， 显 示 数 据 库 结 果 集 中 的 方法 还 有 fetch_assocO0、mysqli_fetch_ 
assoc()、fetch TowO、mysqli_fetch_ row0， 这 些 方法 产生 的 返回 值 和 MySQL 包 的 一 样 。 


11.5.3 ”多 个 查询 


新 的 MySQLI 扩展 提供 了 一 个 新 特性 , 可 以 在 PHP 程序 中 连续 执行 多 个 查询 , 然后 在 适当 的 时 
候 获取 每 个 查询 的 结果 ， 这 时 需要 使 用 一 些 新 的 方法 ， 其 详细 信息 如 表 11-12 所 示 。 


表 11-12 多 个 查询 函数 


名 称 对 象 格式 
mysqli_ multi query0 “| boolean mysqli multi query 连续 执行 多 次 查询 , 是 基于 | class mysqli{ 
(mysqli link,string query) boolean 
multi_query(string query): 
》 
mysqli_ more TesultsO0 | boolean mysqli more results 主要 确定 返回 的 结果 集中 | class mysqli{ 
(mysqli link) 是 否 还 有 其 他 的 结果 集 boolean more_results() 
3 


Imysqli next_resultO boolean mysqli_next result 获取 下 一 个 结果 集 class mysqli{ 
(mysqli link) boolean next resultO} 


下 面 创建 一 个 案例 ， 演 示 在 PHP 程序 中 执行 多 个 查询 。 打 开 记事 本 ， 输 入 下 列 代码 : 
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案例 11-11 
<?php 
$link=mysqli connect ("localhost","root","root"); 
if(!$1link) 
echo mysqli_connect_error () ." 所 在 位 置 为 ".mysqli _connect errno(); 
$link->select_db("com") or die(" 找 不 到 该 数据 库 "); 
$query="select * from cp;"; 
$query.="select cp name from cp"; 
if ($link->multi query ($query)){ 
do{ 
$result=$1link->store result(); 
while ($row=$result->fetch row()) 
{echo "$row[0] ,$row[1] <br>";} 
if($link->more results()) 
{echo "************<br>" >} 
}while ($1link->next result()); 
} 
$1link->close (); 
= 


将 上 述 代 码 保 存 ,文件 名 为 multiphp。 打 开 正 浏览 器 ,在 地 址 栏 中 输入 http://localhost:8080/Data/ 
multi.php， 单 击 【 转 到 】 按 钮 ， 会 显示 如 图 11-15 
所 示 的 窗口 。 

在 本 案例 中 ， 执 行 了 两 次 查询 ， 注 意 在 两 个 
查询 语句 之 间 用 “;” 隔 开 。 在 下 面 直接 使 用 了 
multi_query($query) 方 法 执行 多 语句 查询 ， 并 把 查 
询 的 结果 存储 到 缓存 中 ， 然 后 使 用 “ $result= 
$link->store_result() ”语句 把 存储 的 数据 读 取 出 
来 , 以 数组 的 形式 接收 该 结果 集 。 使 用 fetch_ rowO 
方法 把 结果 集中 的 数据 读 取出 来 并 输出 。 然 后 用 图 11-15 多 次 查询 
方法 more_results0 判 断 下 面 是 否 还 存在 结果 集 ， 

如 果 存 在 则 输出 一 个 分 隔 符 ， 并 开始 执行 下 一 次 循环 。 
上 述 代码 中 的 粗 体 部 分 ， 可 以 使 用 下 列 代码 代替 ; 
if (mysqli_ multi_query(S$1ink,Squery) ){ 
do 
Sresult=mysqli store result ($1link); 
while (Srow=mysqli_fetch_row(Sresult) ) 
{echo "Srow[0],Srow[1] <br>";} 


if(mysqli more results ($1ink)) 
{eChO "六 六 六 六 六 坟 站 本 本 本 本 <D 工 > "> 上 


}while (mysqli next result ($1ink)); 


} 
11.5.4 ”准备 语句 


在 PHP 程序 中 ， 通 常会 重复 执行 一 个 SQL 语句 ， 如 插入 数据 或 修改 数据 等 操作 。 如 果 使 用 传 


统 的 mysql_query0 函 数 执行 ， 每 次 都 要 在 MySQL 服务 端 编译 该 SQL 语句 。 执 行 1000 次 插入 ， 就 
要 编译 1000 次 ， 并 做 1000 次 的 用 户 验 证 ， 并 且 这 些 SQL 语句 几乎 相同 。 这 样 就 会 加 重 MySQL 服 
务 器 的 开销 。 在 MySQL 4.1 中 引入 了 准备 语句 (prepared statement)， 它 可 以 用 低 得 多 的 开销 和 更 少 
的 代码 实现 上 述 任务 。PHP 中 有 以 下 两 种 准备 语句 : 
。 绑 定 参数 ” 绑 定 参数 准备 语句 允许 把 查询 存储 在 MySQL 服务 器 上 ， 只 将 迭代 数据 重复 地 发 
送 给 服务 器 ， 再 将 这 些 迭 代数 据 集成 到 查询 中 执行 。 
。 绑 定 结 果 ” 绑 定 结果 准备 语句 允许 将 PHP 变量 绑 定 到 所 获取 的 相应 字段 ， 从 而 使 用 有 时 难 
以 处 理 的 索引 数组 或 关联 数组 从 结果 中 提取 值 ， 然 后 在 必要 时 使 用 这 些 变量 。 
在 执行 准备 语句 时 ， 需 要 用 到 的 函数 名 称 及 功能 如 表 11-13 所 示 。 


表 11-13 ”准备 语句 函数 


函数 名 称 语法 格式 参数 描述 功能 说 明 
mysqli_stmt_prepareO) boolean mysqli_stmt prepare stmt 表示 一 个 准备 | 准备 要 执行 的 语句 
(mysql_stmt stmt) 语句 对 象 


prepare0 |booleanpreparec0 | | 准备 要 执行 的 语句 


mysqli_stmt_execute() boolean mysqli_stmt execute stmt 表示 一 个 准备 | 执行 准备 语句 ， 该 语句 何 
(mysqli_stmt stmt) 语句 对 象 时 执行 取决 于 语句 类 型 


eecte0 [ooreamceued | [mr 


mysqli_stmt_close() boolean mysqli_stmt_close 同上 关闭 准备 语句 占用 的 资源 
(mysql_stmt stmt) 
close0 [bookeancloseo | |RE 


mysqli_stmt_bind param() | boolean mysqli_ stmt bind param stmt 表示 一 个 准备 | 将 变量 名 绑 定 到 相应 的 字 
(mysqli_stmt, stmt string types, 语句 对 象 , types 参 | 段 ( 绑 定 参数 ) 
mixed & var [mixed &arN]) 数 表示 其 后 各 个 变 
量 (varl...varn) 的 
数据 类 型 


bind_param0 boolean bind_param(string types. 同上 同上 
mixed &var[.&varn]) 

mysqli_stmt_bind result() | boolean mysqli_ stmt_bind result stmt 表示 一 个 准备 | 将 变量 绑 定 到 所 获取 的 字 
(myslqi_stmt, stmt string types. 语句 对 象 , var 表示 | 段 上 〈 绑 定 结果 ) 
mixed &var [.mixed &vam]) 每 个 变量 

boolean bind_result0 boolean bind_result(mixed &var var 表示 变量 将 变量 绑 定 到 所 获取 的 字 
[.mixed & vam]) 段 上 〈 绑 定 结 果 ) 


绑 定 参数 的 示例 如 下 所 示 : 


<?php 
$link=mysqli_ connect ("localhost", "root", "root"); 
if(!$1ink) 
echo mysqli connect error(). "所 在 位 置 为 " mysqli connect errno(); 
$link->select_db("com") or die(" 找 不 到 该 数据 库 ") ; 
$query="insert into cp values(?,?,2,2)"; 
$stmt=$link->stmt init(); 


二 溃 
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在 上 述 代码 中 ,“? ”号 代表 一 个 占 位 符 ， 即 后 面 要 使 用 产品 的 产品 号 、 产 品名 称 、 产 品 价格 、 
产品 说 明 等 。 调 用 bind_param0 方 法 ， 以 方法 中 出 现 的 相同 顺序 ， 将 变量 id、name、price、des 绑 定 
到 问号 表示 的 字段 占 位 符 。 此 查询 准备 好 并 发 送 给 服务 器 ， 此 时 每 个 数据 行 都 已 准备 好 ， 使 用 
execute0 方 法 发 送 给 服务 器 处 理 。 最 后 ， 关 闭 并 回收 占用 资源 。id 等 变量 的 值 都 是 从 另外 一 个 PHP 
页 面 提交 过 来 的 。 

准备 语句 绑 定 结果 的 示例 如 下 所 示 : 


在 上 述 代码 中 ， 查 询 语 句 中 字段 个 数 要 和 “S$stmt->bind_ result($name,$price)” 绑 定 参数 的 个 数 
相同 。fetch0 方 法 主要 获取 准备 语句 结果 的 每 条 记录 ， 并 将 相应 字段 赋 给 绑 定 结果 。 


11.5.5 ”事务 处 理 
事务 是 数据 库 的 实现 ， 而 非 语 言 的 实现 ， 语 言 的 实现 只 不 过 是 包装 了 数据 库 提供 的 API 或 者 直 


接 使 用 数据 库 提供 的 事务 相关 的 SQL Statement。 在 PHP 5.0 中 可 以 调用 MySQL 的 API 来 实现 事务 
的 处 理 。 在 PHP 中 主要 通过 下 列 函 数 实现 事务 的 处 理 ， 分 别 为 autocommit0、commitO、rollbackO。 
其 详细 信息 如 表 11-14 所 示 。 

表 11-14 ”事务 处 理 函 数 


名 称 语法 格式 功 能 对 象 格式 ee 
autocommit() | boolean autocommit(boolean mode) | 控制 MySQL 自动 提交 模 | class mysqli { 
式 的 行为 bool autocommit (bool Smode ) 
} 
commitO boolean commitO 将 当前 事务 提交 给 数据 | class mysqli { 
库 ， 成 功 时 返回 True， 否 | bool commit (void ) 
则 返回 False } 第 
roolbackO boolean roolbackO 回 滚 当前 事务 ， 成 功 时 返 | class mysqli { 项 
回 Tmue， 否 则 返回 False | bool rollback (void ) 这 
} 卫 
下 
事务 的 常用 示例 如 下 所 示 : 和 
数 
<?php 
访 


$link=mysql connect ("localhost","root","root"); 
$success=true; 
$link->autocommit (false); 
…// 对 程序 的 判断 
if($success){ 
$link->commit (); 
echo "事务 成 功 执行 "; 
} 
elset 
$link->rollback (); 
echo "该 事务 出 现 了 一 个 错误 "; 
} 
$link->autocommit (true); 
mysql close(); 
> 


在 上 面 的 示例 中 ， 只 是 列 出 了 事务 在 PHP 程序 中 的 使 用 方式 。 这 里 需要 注意 的 是 , 在 事务 的 每 


个 步骤 执行 之 后 都 会 检查 查询 状态 和 受 影响 的 记录 。 如 果 在 任何 时 刻 失败 ,success 都 将 设置 为 False， 
所 有 的 步骤 都 会 在 脚本 结束 时 回 滚 。 


11.6 PHP 使 用 ODBC 数据 源 


前 面 介绍 了 使 用 PHP 操作 MySQL 数据 库 中 的 数据 ， 如 查询 、 添 加 、 删 除 等 。 如 果 在 PHP 程序 
的 创建 过 程 中 ， 要 求 使 用 另外 的 数据 库 ， 如 SQL Server 2005 或 Access 等 ， 这 时 就 需要 了 解 PHP 的 
另外 一 个 函数 库 一 一 ODBC 函数 库 。 该 函数 库 支持 通过 ODBC 数据 源 连接 其 他 的 数据 源 。 
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11.6.1 连接 指定 数据 库 


PHP 在 Windows 操作 系统 中 ， 不 但 可 以 操作 MySQL 数据 库 ， 还 可 以 通过 Microsoft 公司 提供 
的 ODBC 数据 源 操作 其 他 的 数据 库 。 这 里 可 以 把 PHP 程序 作为 一 个 对 象 看 待 ， 将 存放 数据 的 数据 
库 作 为 一 个 对 象 看 待 ， 在 两 个 格式 不 同 的 对 象 之 间 进行 数据 的 读 取 和 写 入 ， 需 要 一 个 中 间 支 持 者 ， 
这 就 是 数据 库 的 驱动 程序 。 通 过 数据 库 的 驱动 程序 ， 可 以 将 PHP 程序 中 的 命令 编码 翻译 为 数据 库 管 
理 系统 所 识别 的 命令 编码 ， 数 据 库 管理 系统 处 理 完 命令 后 ， 将 执行 结果 返回 。 我 们 知道 ，PHP 中 藤 
入 了 MySQL 数据 库 的 驱动 程序 ，PHP 程序 操作 MySQL 数据 库 是 非常 简单 的 。 如 果 PHP 程序 要 操 
作 Access 数据 库 ， 或 者 SQL Server 2005 数据 库 ， 就 需要 一 个 数据 库 驱 动 程序 。PHP 不 可 能 为 每 一 
个 数据 库 都 提供 相应 的 驱动 程序 和 函数 库 。 一 个 很 好 的 解决 方法 是 ,PHP 通过 ODBC 数据 源 对 这 些 
数据 库 进行 操作 。 

ODBC 数据 源 是 数据 的 来 源 和 访问 该 数据 所 需 的 连接 信息 。 数 据 来 源 的 示例 有 Microsoft 
Access、Microsoft SQL Server、Oracle RDBMS、 电 子 表 格 以 及 文本 文件 。 连 接 信息 的 示例 包括 服务 
器 位 置 、 数 据 库 名 称 、 登 录 太 、 密 码 和 各 种 各 样 描述 如 何 连接 到 数据 源 的 ODBC 驱动 程序 选项 。 
实际 上 可 以 把 ODBC 数据 源 看 作 一 个 组 件 , 该 组 件 中 封装 了 多 种 类 型 的 驱动 程序 PHP 程序 可 以 用 
ODBC 组 件 的 其 中 一 个 驱动 程序 来 操作 相应 的 数据 库 。ODBC 是 一 个 驱动 程序 管理 器 ， 管 理 着 不 同 
类 型 的 驱动 。 

PHP 提供 一 个 函数 库 ， 即 ODBC 函数 库 。 通 过 ODBC 函数 库 操作 相应 的 数据 库 ， 如 Access 数 
据 库 。 在 PHP 的 ODBC 函数 库 中 ， 封 装 了 40 多 种 函数 ， 用 来 完成 对 数据 库 的 操作 。 这 些 函 数 的 名 
称 和 前 面 学 习 MySQL 函数 库 的 名 称 非常 相似 ， 读 者 能 够 非常 容易 地 掌握 这 些 函 数 的 使 用 方式 。 

使 用 ODBC 数据 源 操 作 数据 库 ， 其 步骤 和 前 面 操作 数据 库 的 形式 一 致 ， 首 先 应 该 创建 一 个 数据 
库 连 接 ， 在 连接 的 基础 上 完成 其 他 操作 。 在 创建 数据 库 连 接 之 前 ,需要 在 Windows 系统 中 创建 一 个 
ODBC 数据 源 ， 用 来 指向 特定 的 数据 库 。 数 据 源 的 创建 方式 和 其 他 语言 创建 ODBC 数据 源 的 方式 一 
样 ， 如 VC 等 ， 这 里 不 再 介绍 。ODBC 函数 库 中 用 来 完成 数据 连接 的 函数 分 别 为 odbc_connectO 和 
odbc close0 函 数 。 使 用 ODBC 连接 数据 库 的 示例 如 下 所 示 : 


区 多 
$odbcDsn="liuyan™"; 
$odbcUser=""; 
S$odbcPass=""; 
$conn=odbc_connect ($0dbcDsn, S$odbcUser, S$odbcPass); 
echo "ok"; 
odbc close($conn); 
Ss 


在 上 述 代 码 中 ，odbcDsn 表示 数据 源 名称 ， 该 数据 源 名 称 是 在 Windows 操作 系统 中 创建 的 。 
odbcUser 表示 打开 数据 库 的 用 户 名 称 ，$odbcPass 表示 打开 数据 库 的 用 户 密码 。 函 数 odbc_connectO 
表示 打开 一 个 连接 , 该 连接 需要 相应 的 参数 ; 数据 源 名 称 、 用 户 名 称 和 用 户 密码 等 odbc_close($conn) 
函数 表示 关闭 一 个 数据 库 连 接 。 


11.6.2 ”执行 数据 库 操作 


当 创 建 好 数据 库 连 接 之 后 , 就 可 以 对 数据 库 进行 操作 了 ， 如 获取 数据 库 中 的 数据 、 更 新 数据 等 。 
在 PHP 中 ,对 数据 库 的 操作 同样 也 分 为 两 类 ， 更 新 数据 (添加 、 删 除 、 修 改 ) 和 查询 数据 等 。 使 用 
ODBC 函数 库 操作 数据 库 中 表 的 数据 ， 通 常 需要 下 面 几 个 函数 ， 其 详细 信息 如 表 11-15 所 示 。 


函数 名 称 


表 11-15 ”常用 ODBC 函 数 
语法 格式 


函数 说 明 


odbc_connectO 


resource odbc_connect ( string $dsn，string $user, 
string $password [, int $cursor type] ) 


以 指定 的 数据 源 名 称 、 用 户 名 称 、 
用 户 密码 创建 一 个 连接 


odbc closeO Void odbc close ( resource $connection id ) 关闭 指定 的 数据 库 连接 11 
odbc_exec0 Tesource odbc_exec ( resource $connection id, string | 执行 一 个 query_string 语句 ， 并 返 章 
Squery_string [. int $flags] ) 回 记录 集 
odbc do0 resource odbc do ( resource S$conn id， string | 在 指定 的 连接 执行 query 语句 ， 其 总 
S$query) 结果 是 一 个 记录 集 指针 
odbc_fetch rowO bool odbc fetch row ( resource $result id [，int | 在 数据 库 的 记录 集中 获取 一 行 。 如 下 
Srow_number] ) 果 结 果 集 中 有 记录 , 返回 True, 否 
则 返回 False。 访 
odbc_resultO mixed odbc result ( resource Sresult id，mixed | 返回 指定 记录 集中 指定 字段 的 值 
$field ) 
odbc errorO string odbe_error ( [resource $connection id] ) 以 字符 串 的 形式 返回 报错 消息 
odbc_field_ name() string odbc field name ( resource Sresult id，int | 返回 指定 字段 的 名 称 


$field number ) 


1. 更 新 数据 

在 PHP 程序 中 ， 对 Access 数据 库 或 者 其 他 数据 库 执行 操作 前 ， 应 保证 数据 库 连 接 成 功 ， 这 样 
才 可 以 执行 添加 、 删 除 或 修改 等 操作 。 通 过 ODBC 数据 源 完 成 上 面 的 操作 ， 需 要 用 到 函数 
odbc_execO0， 该 函数 的 主要 作用 是 执行 相应 的 SQL 语句 。 现 在 以 Access 数据 库 为 例 ， 向 数据 库 中 
添加 数据 。 其 示例 如 下 所 示 : 


区 

$odbcDsn="liuyan"; 

S$odbcUser=""; 

$odbcPass=""; 

$sql="insert into user values ('guest','123')"; 
$conn=odbc connect($odbcDsn, S$odbcUser, S$odbcPass); 
$query =odbc exec($conn, $sql); 

echo "操作 执行 成 功 "; 

$conn.odbc close($conn); 

2 


在 上 述 代 码 中 ，odbc_exec ($conn,$sql) 表示 在 打开 的 连接 conn 上 执行 SQL 语句 。 

2. 获取 并 显示 数据 

在 PHP 页 面 上 显示 数据 库 中 的 数据 是 一 个 非常 重要 的 操作 。 下 面 以 Access 数据 库 为 例 ， 创 建 
一 个 案例 ， 演 示 获 取 数 据 库 中 的 数据 。 创 建 一 个 Access 数据 库 od， 并 创建 指向 该 数据 库 的 数据 源 ， 


名 称 为 liuyan。 打 开 记 事 本 ， 输 入 以 下 内 容 : 
案例 11-12 


Es 
$odbcDsn="liuyan™"; 
$odbcUser=""; 
$odbcPass=""; 
$conn=odbc_connect ($0dbcDsn, S$odbcUser, S$odbcPass); 
$sql="select * from user"; 
$result id=odbc do($conn,$sql); 
while(odbc fetch row(Sresult id)) 


$AAl = odbc result ($result idqd, J 
$AA2 = odbc result ($result id, 2) 
echo $AAl." ".$AA2."<br/>\n"; 

} 

odbc close($conn); 

二 


将 上 述 代 码 保存 , 文件 名 为 odbccon.php。 打开 正 浏览 器 , 在 地 址 栏 中 输入 http://localhost:8080/ 
Data/odbccon php， 单 击 【 转 到 】 按 钮 ， 会 显示 如 图 11-16 所 示 的 窗口 。 


卫 
ud 
5 
十 
[二 
四 
[3 
Li 
完 
全 
学 
习 
手 
册 


文件 岛 辑 妈 音 看 W) 改 功 人 工具 中。 条 勒 D 
EP /css wo Ds csc lz) 


admin admin 
| 

guest 123 
Eucst 123 


图 11-16 数据 显示 


在 上 述 代码 中 ， 创 建 了 一 个 数据 库 连 接 对 象 conn， 使 用 函数 odbc_do0 返 回 一 个 记录 集 对 象 
result id， 并 使 用 odbc_fetch rowO 函 数 获取 记录 集 对 象 中 的 每 一 条 记录 ， 如 果 记 录 集 中 包含 数据 库 
记录 则 返回 True， 和 否则 返回 False。 在 while 循环 中 ， 使 用 odbc_result0 函 数 显 示 指 定 字段 的 值 。 
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第 12 章 PEAR 


《学习 目标 |owiecuve 


PEAR 实际 上 是 PHP 的 第 三 方 类 库 ， 是 PHP 扩展 包 ， 其 中 包含 了 大 量 有 用 的 类 ， 如 校 验 信息 
格式 、 构 建 HTML 表单 等 ,在 PHP 程序 中 使 用 PEAR 包 , 是 优秀 PHP 程序 员 的 第 一 选择 。 使 用 PEAR 
包 可 以 达到 优秀 代码 重用 的 目的 , 一 个 大 型 的 Web 项 目 可 以 由 多 个 PEAR 包 组 成 , 实现 功能 的 相互 
分 离 。 本 章 将 介绍 PEAR 包 的 概念 和 使 用 ， 以 及 常用 的 PEAR 包 的 使 用 。 


ER 
2 内 容 摘要 |avstrsct 


了 解 PEAR 包 的 概念 和 优点 

掌握 PEAR 管理 的 安装 和 更 新 
熟练 掌握 使 用 PEAR 管理 PEAR 包 
掌握 PEAR 的 常用 命令 

掌握 HITML_ QuickForm 包 创 建 表单 
掌握 Calendar 包 获取 时 间 及 日 期 
掌握 使 用 AUTH 认证 

掌握 使 用 PEAR 包 执 行文 件 的 上 传 


12.1 PEAR 概述 


PHP 是 一 种 非常 优秀 的 脚本 语言 ， 简 洁 、 高 效 ， 随 着 PHP 5.0 的 发 布 ， 越 来 越 多 的 人 使 用 它 进 
行动 态 网 站 的 开发 ，PHP 已 经 成 为 最 优秀 的 Internet 开发 语言 之 一 ， 尤 其 对 于 那些 需要 快速 、 高 效 
地 开发 中 小 规模 的 商业 应 用 的 网 站 开发 人 员 来 说 ,PHP 是 首选 语言 。 但 是 随 着 PHP 应 用 的 不 断 增多 ， 
对 于 这 些 应 用 缺乏 统一 的 标准 和 有 效 的 管理 ， 因 此 ，PHP 社区 很 难 像 Perl 社区 的 人 们 那样 方便 地 共 
享 彼此 的 代码 和 应 用 , 因为 PHP 缺乏 像 CPAN 那样 的 统一 的 代码 库 来 分 类 管理 应 用 的 代码 模块 ( 熟 
悉 Perl 的 人 都 知道 ，CPAN 是 一 个 巨大 的 Perl 扩展 模块 仓库 ,编写 的 应 用 模块 可 以 放 在 CPAN 下 面 
的 适当 分 类 目录 下 面 ， 其 他 人 可 以 很 方便 地 复 用 ， 当 然 ， 编 写 应 用 模块 时 也 需要 遵守 其 中 的 准则 )。 
为 此 ，PEAR 就 应 运 而 生 了 ， 并 且 从 4.04 开始 ， 随 着 PHP 核心 一 起 发 布 。 

PEAR 是 PHP 的 官方 开源 类 库 ， 是 PHP Extension and Application Repository 的 缩写 。PEAR 将 
PHP 程序 开发 过 程 中 常用 的 功能 代码 编写 成 类 库 ， 涵 盖 了 页 面 显 示 、 数 据 库 访问 、 文 件 操作 、 数 据 
结构 、 缓 存 操作 、 网 络 协议 等 许多 方面 ,用 户 可 以 很 方便 地 使 用 .PEAR 的 大 部 分 Package 采用 LGPL、 
PHP、BSD 许可 证 ， 可 以 自由 地 使 用 源码 。 

PEAR 是 为 PHP 代码 的 重用 而 开发 的 ， 使 用 PEAR 可 以 大 大 提高 PHP 程序 开发 效率 。 前 人 已 
经 完成 的 工作 , 可 以 直接 使 用 , 不 需要 重复 开发 , 更 可 以 保证 开发 代码 的 质量 。 在 PHP 中 使 用 PEAR 
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有 下 面 几 个 优点 ， 如 表 12-1 所 示 。 

表 12-1 PEAR 优 点 

优点 说 了 明 

许多 程序 代码 都 被 撰写 在 类 库 中 ， 开 发 者 只 需 引 用 包含 实现 功能 的 代码 库 
的 PHP 文档 ， 就 可 以 拥有 类 库 所 提供 的 功能 ， 省 去 自行 撰写 的 时 间 与 精力 
由 于 这 些 类 库 套件 (PEAR 包 不 同 的 版 本 ) 都 将 原始 码 公开 ， 使 用 者 若 发 现 
bug， 可 以 立即 向 套件 的 开发 者 反映 ， 开 发 者 通常 也 会 立即 加 以 修正 ， 让 类 
库 的 错误 减 到 最 小 
自行 撰写 的 代码 与 PEAR 类 库 的 代码 分 开 ， 若 PEAR 套件 因为 有 新 功能 被 
开发 或 修正 错误 而 释 出 新 版 本 ， 只 需 更 新 套件 即 可 ， 完 全 不 会 影响 到 自行 
撰写 的 程序 代码 
PEAR 套件 提供 的 功能 非常 多 ， 如 资料 库 的 连结 、 设 定 档 的 处 理 、 身 份 的 认 
证 以 及 表单 的 处 理 等 
PEAR 提供 许多 模板 的 类 库 , 可 让 程序 开发 与 版 面 设计 分 开 , 便于 程序 开发 
者 与 网 页 设计 者 分 工 合作 
类 库 的 注解 说 明 十 分 详细 ， 程 序 代码 的 撰写 具有 一 定 的 规则 与 格式 ， 而 且 
完全 物质 化 ， 让 开发 者 容易 看 懂 并 学 习 。 有 些 套件 还 内 附 教学 文件 与 范例 
文档 ， 最 重要 的 是 这 些 类 库 及 说 明文 档 都 是 免费 的 


在 进一步 了 解 PHP 之 前 ， 必 须 清楚 两 个 名 词 ， 以 免 读 者 混淆 ， 一 个 是 “程序 库 ”， 另外 一 个 是 
“套件 ”。 假 如 一 个 PHP 开发 者 ， 编 写 了 一 个 可 以 处 理 所 有 数据 库存 取 的 “程序 库 ” 并 想 将 这 个 程 
序 库 加 入 PEAR 分 享 给 全 世界 使 用 。 当 撰写 完 数据 库 的 程序 库 后 ， 必 须 按照 PEAR 的 规定 ， 将 程序 
库 压缩 成 一 个 “ 件 ” 档 ， 放 在 PEAR 官方 网 站 供 所 有 人 下 载 ， 也 由 于 遵循 了 PEAR 的 规定 来 压缩 文 
件 , 这 个 套件 就 可 以 用 PEAR 的 套件 管理 指令 来 安装 或 管理 , 这 对 PEAR 使 用 者 而 言 是 相当 方便 的 ， 
关于 套件 管理 指令 会 在 下 一 节 中 介绍 。 

PEAR 截至 目前 为 止 释 出 的 套件 已 高 达 384 个 ， 共 分 为 35 大 类 , 而且 在 不 断 增加 中 。 目 前 常用 
的 包 如 表 12-2 所 示 。 
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表 12-2 常用 PEAR 包 


包 名 说 明 
Archive_Tar 这 个 包 有 利于 管理 tar 文件 ， 提 供 了 创建 、 列 举 、 提 取 和 增加 tar 文件 的 方法 。 此 外 ， 假 如 
安装 了 相应 的 PHP 扩展 ， 它 还 支持 Gzip 和 Bzip2 压缩 算法 。PEAR 必须 有 这 个 包 才 能 正常 
运行 


Console_Getopt 如 果 能 在 执行 时 提供 选项 通过 命令 行 修改 脚本 执行 的 行为 ， 往 往 是 很 有 用 的 。 例 如 ， 可 以 
使 用 pear 命令 传递 -v 来 验证 PEAR 所 使 用 的 版 本 。 另 外 ， 该 包 为 读 取 这 些 选项 提供 了 一 种 
标准 方法 ， 如 果 提 供 的 语法 不 符合 预定 义 的 规范 ， 它 还 会 为 用 户 提供 报错 消息 。PEAR 包 必 


须 有 这 个 包 才 能 正常 运行 

DB DB 包 为 抽象 的 数据 库 层 的 通信 提供 了 面向 对 象 的 查询 API。 将 应 用 程序 和 数据 库 之 间 的 移 
植 提供 了 便利 

Mail 电子 邮件 类 提供 与 电子 邮件 相关 的 套件 

Net_Socket 用 来 简化 对 TCP 套 接 字 的 管理 ， 它 为 了 完成 连接 ， 在 套 接 字 之 间 读 取 和 写 入 信息 提供 了 统 
一 的 API 

Net_ SMTP 提供 了 SMTP 协议 的 实现 ， 能 够 完成 以 下 任务 ， 包 括 连 接 SMTP 服务 器 、 断 开 连 接 、 完 成 
SMTP 的 认证 、 鉴 别 发 送 者 以 及 发 送 邮件 等 

PEAR PEAR 需要 这 个 包 才 能 正常 运行 


人 说 了 明 

PHPUnit 单元 测试 是 为 确保 代码 块 正常 操作 的 一 种 测试 方法 a 
XML Parser | 为 解析 XML 文件 提供 了 面向 对 象 的 简单 解决 方案 : 
XML RPC | XML RPC 协议 基于 PHP 的 实现 ， 该 协议 是 在 因特网 上 远程 调用 过 程 的 一 种 方法 oe 
Auth | 这 个 包 能 通过 很 多 机 制 方便 用 户 认证 ， 
HTML QuickForm | 此 包 有 利于 HTML 表单 的 创建 、 生 成 和 验证 

Log 这 个 包 提供 了 一 个 抽象 的 日 志 工 具 ,支持 记录 到 控制 台 、 文 件 、SQL、SQLite、 邮 件 和 mcal 


除了 上 面 介绍 的 这 些 包 之 外 ， 还 提供 了 其 他 的 PEAR 包 ， 读 者 可 以 在 http://pear.php.net/ 
packages.php 网 站 查看 最 新 、 最 全 的 PEAR 包 。 


12.2 ”PEAR 管理 器 安装 和 更 新 


上 一 小 节 介 绍 了 PEAR 包 具 有 代码 重用 的 好 处 , 并 且 众 多 的 PHP 爱好 者 又 提供 了 大 量 的 PEAR 
包 ， 供 程序 员 在 开发 PHP 程序 时 使 用 ， 那 么 PEAR 包 怎 么 使 用 ，PEAR 包 如 何 进行 管理 ，PEAR 包 
如 何 进行 升级 呢 ? 本 节 将 首先 对 PEAR 包 的 安装 进行 介绍 。 


YHV3d 贡 总 站 


12.2.1 PEAR 管理 器 安装 


PEAR 的 产生 对 于 快速 编写 PHP 程序 起 到 了 决定 性 作用 ， 以 至 于 在 PHP 4.3.0 后 的 版 本 ， 发 行 
包 中 都 包括 了 一 个 稳定 的 PEAR 包 。 在 PHP 5.0 中 ， 如 果 使 用 的 PHP 安装 文件 是 zip 压缩 包 ， 会 在 
PHP 的 安装 目录 C:\Web\php 下 自动 生成 一 个 PEAR 文件 夹 ， 后 面 使 用 到 的 PEAR 包 都 是 放 在 该 目 
录 下 ， 在 该 文件 夹 下 会 存在 一 个 go-pear.phar 的 文件 。PEAR 的 安装 过 程 如 下 所 示 : 

(1) 将 PHP 的 安装 目录 C:\WWeb\php\ 
PEAR 文件 夹 下 的 go-pearphar 文件 改 成 
go-pearphp 文件 。 

(2) 单 击 【开始 】 菜 单 ， 选 择 【 运 行 】 
命令 ， 在 弹出 的 对 话 框 中 输入 “cmd”， 在 命 
令 提 示 符 窗口 中 进入 PEAR 的 目录 ， 如 图 
12-1 所 示 。 

(3) 在 该 目录 的 下 面 输入 命令 “php 
go-pear.php”， 然 后 按 回 车 键 , 会 显示 如 图 
12-2 所 示 的 信息 

在 图 12-2 中 会 提示 是 是 否 安装 PEAR， 如 
果 安 装 ， 则 按 回 车 键 ， 会 在 窗口 中 显示 如 图 
12-3 所 示 的 信息 。 图 12-2 安装 PEAR 


图 12-1 进入 PEAR 目录 


在 图 12-3 中 显示 是 选择 安装 还 是 全 部 安 
装 ， 其 中 ，Installation base 表示 安装 PEAR 的 
根 目录 ; ”Binaries directory 表示 程序 和 PEAR 
包 中 PHP 脚本 安装 的 地 方 。PEAR 将 在 这 里 执 
行 。 PHP code directory 表示 PHP 代码 的 安装 位 
置 。 这 个 路 径 必 须 在 php.ini 的 include_path 
© 包含 Documentation directory 表示 文档 的 基本 
目录 ,默认 情况 下 是 $php_dir/doc， 每 一 个 包 的 
文档 作为 $doc_dr/Package/file 来 安装 ; Data 
directory 表示 PEAR 安装 程序 保存 数据 文件 的 
地 方 。Tests directory 表示 包 旧 的 测试 脚本 安装 
的 地 方 ， 包 的 名 字 也 添加 到 这 个 路 径 。 实 际 上 
在 这 里 ， 要 么 全 部 安 要 么 不 安装 。 这 里 直 
回 车 键 ， 使 用 全 部 安装 ， 会 在 命令 提示 符 
1 显示 如 图 12-4 所 示 的 信息 。 
在 图 12-4 中 显示 的 是 ，PEAR 安装 基本 的 
J 包 。 当 安装 完成 后 ， 会 自动 在 命令 提示 
1 中 显示 如 图 12-5 所 示 的 信息 。 
安装 完成 之 后 , 为 了 使 PHP 程序 能 够 识 图 12-4 PEAR 包 安装 
别 和 到 相应 雏 PEAR 包 , 需要 在 php.ini 文件 
中 写 入 一 些 信息 。 按 回 车 键 ， 会 在 窗口 中 显示 
如 图 12-6 所 示 的 信息 。 
图 12-6 中 显示 的 是 在 php.ini 中 要 写 入 的 
相关 人 继续 按 回 车 键 , 会 把 这 些 信 息 写 入 。 
打开 php.inp， 其 信息 如 下 所 示 : 
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了 站 站 站 站 让 Added by go-pear 
include path=".;C:\Web\php\PEAR\ 


pear" 
7 刘 率 半音 


图 12-5 写 入 php.ini 
写 入 信息 后 ， 表 示 安 装 全 部 完成 ， 
令 提示 符 窗 口中 显示 如 图 12-7 所 示 的 信息 。 
(4) 安装 完成 后 ， 打 开 C:\WWeb\php\PEAR 
文件 夹 ， 如 图 12-8 所 示 。 
安装 结束 时 , 会 创建 如 图 12-8 所 示 的 文件 
和 文件 夹 ， 除 了 go-pearphp 外 ， 还 有 一 个 注册 
表 文 件 PEAR_ENVreg。 执 行 这 个 文件 将 为 一 
些 PEAR 特定 的 变量 创建 环境 变量 。pear 文件 
夹 中 放置 的 是 最 基本 的 PEAR 包 ，tem 文件 夹 
中 放置 的 是 一 些 临 时 性 文件 。 环 境 参 数 的 自动 
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设 定 只 需 双击 pearbat 文件 即 可 。 
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图 12-7 安装 成 功 图 12-8 PEAR 文件 显示 第 
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PEAR 包 在 不 断 地 进行 改进 ， 也 就 是 说 ， 有 时 需要 检查 并 更 新 系统 ， 这 在 UNIX 和 Windows 平 
台 上 是 一 个 很 简单 的 过 程 ， 只 要 执行 位 于 PHP installation_dinPEAR 目录 中 的 go-pear.php 脚本 就 可 
完成 。 路 径 如 下 所 示 : 

C:\Web\php\PEAR>php go-pear.php 

执行 该 操作 实际 上 是 重新 启动 安装 过 程 ， 覆 六 之 前 安装 的 包 管理 器 版 本 。 

PEAR 安装 包 的 更 新 可 以 从 站 点 go-pear.org 中 获取 ，http://go-pear.org 是 一 个 Web 站 点 , 这 个 站 
点 很 特殊 ， 里 面 只 有 一 个 文件 ， 是 一 个 单独 的 PHP 脚本 ， 可 以 下 载 并 且 执 行 它 。 这 个 文件 将 执行 最 
近 的 稳定 发 行 包 。go-pear 是 交互 平台 ， 可 以 在 服务 器 上 用 命令 行 来 获得 PEAR。 


12.3 使 用 PEAR 管理 器 


安装 了 PEAR 包 管 理 器 ， 就 可 以 通过 PEAR 管理 器 ， 对 PHP 程序 中 需要 使 用 到 的 PEAR 包 进 
行 查看 、 删 除 、 升 级 等 操作 。PEAR 包 管理 器 通过 在 命令 提示 符 窗口 中 ， 输 入 pear 命令 完成 这 些 
操作 。 


12.3.1 查看 PEAR 安装 包 


当 PEAR 管理 器 安装 完成 后 ， 就 可 以 直接 使 用 pear 命令 了 。 该 命令 的 语法 格式 如 下 所 示 : 
pear [options] command [command-options] <parameters> 


为 了 更 好 地 熟悉 PEAR 包 管 理 器 ， 可 以 在 命令 提示 符 窗 口中 ， 直 接 输 入 如 下 所 示 的 命令 : 


这 里 需要 注意 的 是 ， 需 要 进入 C:\Web\php\PEAR 目录 下 执行 。 当 执行 了 该 命令 后 ， 会 看 到 一 组 
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命令 和 相应 的 语法 信息 。 如 果 想 具体 了 解 某 个 命令 的 信息 ， 可 以 使 用 如 下 所 示 的 命令 : 


pear help <command> 


在 pear 文 件 夹 下 可 以 直接 执行 pear 命令 ， 如 果 在 其 他 位 置 执行 pear 命令 ， 则 需要 将 PEAR 目录 增加 
提示 | ”到 系统 路 径 中 。 


1. 查看 系统 安装 包 

OD 查看 当前 系统 上 安装 的 PEAR 包 非常 简 
单 ， 只 需要 执行 pear list 命令 ， 显 示 信息 如 图 
12-9 所 示 。 

2. 查看 包 的 详细 信息 

图 12-9 显示 了 服务 器 上 安装 了 4 个 包 , 这 
些 信息 非常 简单 ， 除 了 包 名 和 版 本 之 外 没有 提 
供 任何 相关 的 信息 。 如 果 要 了 解 PEAR 包 的 详细 信息 ， 可 以 执行 info 命令 ， 并 传递 包 名 给 它 。 例 如 
要 了 解 Console_Getopt 包 的 详细 信息 ， 可 以 执行 如 下 命令 : 


图 12-9 PEAR 包 安 装 信息 


pear info Console Getopt 


其 执行 结果 如 图 12-10 所 示 。 


12.3.2 升级 PEAR 包 
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当 安 装 好 PEAR 包 后 ， 还 需要 不 断 地 维护 
PEAR 包 ， 因 为 这 些 PEAR 包 还 处 在 一 个 开发 
的 阶段 , 可 能 在 PEAR 包 中 会 存在 bug, 因此 需 
要 不 断 地 查看 是 否 有 新 的 版 本 出 现 。 如 果 出 现 
了 新 的 版 本 , 就 需要 对 这 些 PEAR 包 进 行 更 新 。 图 12-10 详细 信息 
升级 PEAR 包 的 命令 格式 如 下 所 示 : 


pear upgrade [package name] 

例如 ， 在 命令 提示 符 窗 口中 输入 如 下 命令 : 

pear upgrade pear 

该 命令 表示 升级 PEAR 包 的 环境 。 如 果 系统 上 安装 的 版 本 是 最 新 版 本 ， 会 在 命令 提示 符 窗口 中 
显示 如 图 12-11 所 示 的 信息 

如 果 系 统 上 安装 的 是 旧版 本 ， 就 会 进行 更 新 。 这 里 需要 注意 的 是 ， 要 保证 该 计算 机 在 Intemet 
上 。 其 显示 信息 如 图 12-12 所 示 。 


EE 


图 12-11 升级 失败 图 12-12 升级 成 功 
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如 果 在 系统 上 安装 的 PEAR 包 太 多 ， 一 个 一 个 地 进行 升级 ， 就 会 有 点 麻烦 ， 这 时 可 以 使 用 命令 
Upgrade-all 进行 升级 。 执 行 如 下 命令 : 
Pear upgrade-all 


执行 该 命令 后 可 能 会 出 现 有 些 包 版 本 和 以 前 的 版 本 不 兼容 的 情况 ， 因 此 不 推荐 使 用 此 命令 ， 除 。。 
非 非常 了 解 升级 每 个 包 之 后 的 结果 。 


12.3.3 安装 PEAR 包 


在 系统 上 安装 PEAR 包 ， 实 际 上 分 为 两 个 步骤 ， 第 一 步 是 了 解 要 安装 的 PEAR 包 ， 第 二 步 是 安 
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装 PEAR 包 。 现 在 以 Validate US 包 为 例 ， 演 示 PEAR 包 的 安装 过 程 。 12 
1. 了 解 PEAR 包 章 
如 果 要 安装 一 个 PEAR 包 ， 首 先 需要 了 解 该 PEAR 包 ， 如 包 的 名 称 和 作用 。 打 开 正 浏览 器 ， J 
在 地 址 栏 中 输入 http://wwwpearphp net， 单 击 【 转 到 】 按 钮 ， 会 显示 如 图 12-13 所 示 的 窗口 。 号 
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图 12-13 PEAR 官方 下 载 网 站 


该 窗口 显示 的 是 PEAR 包 的 官方 网 站 ， 稳 定 的 并 且 最 新 的 开发 包 都 可 以 在 该 网 站 获得 。 在 该 窗 
口中 单 击 List Packages 超级 链接 ， 会 显示 如 图 12-14 所 示 的 窗口 。 
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图 12-14 PEAR 包 选 择 
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在 图 12-14 中 ， 显 示 了 所 有 的 PEAR 包 ， 每 个 链接 后 面 的 数字 表示 该 包 具 有 的 类 别 数量 。 现 在 
以 Validate 包 为 例 演 示 下 载 PEAR 包 。Validate 包 主要 用 在 数据 的 校 验 方面 ， 如 数据 校 验 、 电 子 邮 件 
校 验 。 在 页 面 中 找到 要 安装 的 PEAR 包 ， 单 击 Validate 超级 链接 。 
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图 12-15 选择 Validate US 


在 图 12-15 中 显示 的 是 进行 校 验 时 ， 可 以 用 到 不 同 环境 下 的 套件 。 找 到 要 安装 的 Validate_US， 
单 击 Validate_US 超级 链接 ， 会 显示 如 图 12-16 所 示 的 窗口 。 


PEAR :: Package := Validate US - Wicrosol 
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图 12-16 下 载 Validate US 


在 图 12-16 中 显示 的 是 关于 PEAR 包 的 描述 信息 ， 如 该 包 的 版 本 号 、 是 否 具有 bug 及 该 包 的 应 
用 领域 等 ， 也 可 以 查阅 该 PEAR 包 的 文档 信息 。 

2. 安装 PEAR 包 

记 住 该 包 的 名 称 和 用 途 后 ， 就 可 以 安装 PEAR 包 了 。 安 装 PEAR 包 是 一 个 完全 自动 的 过 程 ， 要 
安装 的 包 不 需要 提前 下 载 ， 直 接 可 以 在 客户 端 使 用 install 命令 下 载 并 安装 。 该 命令 的 语法 格式 如 下 
所 示 : 


pear install [options] package 


下 面 安 装 Validate US 包 ， 在 窗口 中 输入 “pear install -o Validate US-0.5.2” 命 令 ， 其 中 -o 表示 
在 安装 该 包 的 同时 ， 会 安装 该 包 运 行 时 所 需要 
的 其 他 PEAR 包 ， 如 图 12-17 所 示 。 

如 果 PEAR 包 在 安装 时 ， 没 有 为 该 包 安装 
相应 的 依赖 包 ， 则 该 包 的 某 些 功能 可 能 无 法 使 
用 。 如 果 遇 到 安装 没有 成 功 ， 一 般 都 是 缺少 相 
应 的 依赖 包 造 成 的 , 这 时 可 以 使 用 -o 命令 选项 ， 图 1227 .安装 NE US:0.52 
也 可 以 使 用 命令 选项 -a 来 为 PEAR 包 安 装 可 选 和 必需 的 依赖 包 。 其 语法 格式 如 下 所 示 : 

pear install -a Validate US-0.5.2 

可 以 看 出 上 面 的 安装 方式 实际 上 是 一 种 网 络 上 的 安装 , 需要 计算 机 系统 连接 到 Internet, 才能 完 
成 安装 任务 。 这 样 做 可 以 保证 下 载 的 PEAR 包 都 是 最 新 的 ， 但 是 如 果 想 使 用 以 前 的 安装 版 本 ， 就 需 
要 把 旧版 本 下 载 下 来 。 下载 完毕 后 将 文件 解压 , 并 将 文件 放 在 PHP 安装 根 目录 的 PEAR 目录 结构 中 
的 适当 位 置 。 


12.3.4 删除 PEAR 包 


如 果 使 用 一 个 PEAR 包 ， 发 现 没有 另外 的 方案 方便 快捷 或 者 不 再 使 用 这 个 PEAR 包 ， 就 应 该 从 
当前 系统 中 仓 载 该 包 。 该 操作 是 通过 命令 uninstall 实现 的 ， 其 语法 格式 如 下 所 示 : 


pear uninstall [options] package name 
例如 为 了 下 载 Auth 包 ， 可 以 执行 如 下 命令 : 
pear uninstall Auth 


如 果 想 要 了 解 uninstall 命令 的 详细 信息 ， 可 以 使 用 如 下 命令 : 


pear help uninstall 
12.3.5 测试 PEAR 包 


PEAR 包 安装 完成 后 ,就 可 以 被 其 他 的 PHP 程序 调用 。 下 面 创 建 一 个 案例 , 以 安装 的 Validate US 
为 例 ， 演 示 PEAR 包 的 使 用 流程 。 打 开 记 事 本 ， 输 入 下 列 代码 : 


案例 12-1 
<?php 
require once("Validate/US.php"); 
$validate=new Validate US(); 
if(!$validate->phoneNumber ("614-876530932")){ 
echo "电话 号 码 错 误 "; 
Elsel 


echo "电话 号 码 正 确 "; 
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} 

Se 

将 上 述 代 码 保存 ， 文 件 名 为 test.php， 保 存 到 C:\Web\apache\htdocs 目录 下 。 打 开 正 浏览 器 ,在 
地 址 栏 中 输入 http://localhost:8080/test.php, 单 击 【 转 到 】 按 钮 ， 
会 显示 如 图 12-18 所 示 的 窗口 。 

在 上 面 的 代码 中 , 首先 把 需要 使 用 的 PEAR 包 包 含 到 当前 
的 文件 中 , 在 包含 时 要 注 明 该 PEAR 包 的 路 径 信息 。 使 用 语句 
“$validate=new Validate USO” 创 建 了 一 个 validate 对 象 , 用 来 
执行 PEAR 包 中 相应 的 操作 ， 如 调用 phoneNumber0 方 法 校 验 
电话 号 码 的 格式 是 否 正确 。 除 了 上 面 使 用 的 方法 phoneNumber0 ，Validate US 还 具有 方法 
postalCode0 、region0、ssn0 等 ， 分 别 用 来 校 验 ZIP 编码 、 验 证 州 缩写 、 社 会 保险 号 等 。 其 详细 信息 
可 以 查阅 该 PEAR 包 的 帮助 文档 。 


12.4 常用 PEAR 包 


PEAR 包 的 使 用 加 快 了 PHP 程序 开发 ， 实 现 了 数据 处 理 和 数据 实现 的 相互 分 离 。 每 个 程序 员 都 
可 以 开发 出 PEAR 包 并 在 网 络 上 和 别人 共享 ， 在 PEAR 包 的 官方 网 站 上 ， 已 经 有 448 个 PEAR 包 。 
本 节 将 介绍 一 些 常 用 PEAR 包 的 使 用 。 


图 12-18 电话 号 码 校 验 


12.4.1 使 用 HTML_QuickForm 


PHP 作为 一 个 基本 的 Web 语言 , 不 可 避免 地 要 进行 表单 的 操作 ， 如 在 表单 中 显示 数据 、 获 取 客 
户 端 提交 的 数据 等 , 写 一 个 大 型 的 Web 程序 , 要 多 次 用 到 表单 。 在 PEAR 包 中 , 有 一 个 强大 的 PEAR 
包 可 以 完成 表单 的 基本 操作 ， 它 就 是 HIML_QuickForm。 它 能 够 完成 表单 的 创建 、 数 据 的 提交 和 显 
示 ， 而 且 更 有 用 的 是 ， 客 户 端 和 服务 器 端 都 能 够 得 到 验证 ， 即 快速 又 简单 。 

1. 安装 HTML_QuickForm 

安装 PEAR 包 只 需要 两 个 条 件 : PHP 4.2 以 上 ， 并 且 有 HTML Common 包 。 氨 今 为 止 HIML 
QuickForm 2 是 最 新 的 版 本 ， 它 需要 对 应 的 HIML_Common 2。 这 里 使 用 HIML_QuickForm 3.2.9， 
因为 最 新 的 HIML_QuickFomm 2 是 一 个 测试 版 本 ， 可 能 会 存在 bug。 下 面 开始 安装 HIML_Qnuick- 
Form， 打 开 命 令 提 示 符 窗口 ， 输 入 如 下 命令 : 


pear install -oO HTML QuickForm 3.2.9 


按 回 车 键 ， 会 在 命令 提示 符 窗 口中 显示 如 图 
12-19 所 示 的 信息 。 

安装 完成 后 ， 就 可 以 使 用 该 PEAR 包 来 创建 表 
单 、 验 证 表单 了 。 

2. 创建 表单 

HIML QuickForm 安装 完成 后 ， 就 可 以 完成 创 图 12-19 安装 HIML QuickForm 


建 表单 的 操作 了 。 下 面 创建 一 个 案例 ， 演 示 使 用 PEAR 包 HIML QuickForm 创建 表单 。 打 开 记 事 
本 ， 输 入 下 列 代码 : 


案例 12-2 

<HTML> 

<HEAD> 和 
<TITLE> PERR: :HTML QuickForm </TITLE> 

<META NAME="Generator" CONTENT="EditPlus"> 

<META NAME="Author" CONTENT="liuhaisong"> 
</HEAD> 

<BODY> 

ws 

require once("HTML/QuickForm.php"); 

// 建 立 一 个 表单 对 象 

$form = new HTML QuickForm('frmTest', "post')7 

/* 利 用 该 对 象 的 addElement () 方法 增加 4 个 表单 元 素 
aqddElemment () 的 三 个 参数 分 别 表示 类 型 、 名 称 、 显 示 的 文字 
$form->addElement ('header'， 'header'，“' 请 登录 '); 
$form->addElement ('text'， 'name'，' 用 户 名 : '); 
$form->addElement ('password', 'password'， ' 密码 : '); 
$form->addElement ('submit'， 'submit'， ' 提 交 '); 

// 输 出 到 浏览 器 

$form->display (); 

2 

</BODY> 

</HTML> 


将 上 述 代 码 保存 ， 文 件 名 为 Html.php， 并 将 该 文件 保存 在 C:\Web\apache\htdocs\PE 目录 下 。 打 
开 正 浏览 器 ， 在 地 址 栏 中 输 入 
http://localhost:8080/PE/Html.php， 单 击 【 转 到 】 按 钮 ， 会 显示 
如 图 12.20 所 示 的 窗口。 SR 

在 本 案例 中 ， 使 用 require_once 语句 把 需要 使 用 的 PEAR | 请 划 冰 
包 包含 到 当前 的 PHP 文件 中 ， 这 时 就 可 以 直接 使 用 PEAR 包 | 用 户 名 : 
中 的 方法 。 然 后 使 用 *$ftorm = new HTML QuickForm('frmTest', 
post)” 创 建 表单 对 象 formm， 该 表单 对 象 所 指向 的 表单 名 称 为 
frmTest， 表 单 提交 的 方式 是 以 post (数据 块 ) 方式 提交 。 表 单 
存在 后 , 就 可 以 在 这 个 表单 中 添加 相应 的 表单 元 素 了 。 这 里 主 12-20 显示 窗口 
要 使 用 方法 addElemmentO 添 加 相应 的 表单 元 素 ， 该 方法 有 三 
个 参数 ， 分 别 表 示 类 型 、 名 称 、 显 示 的 文字 。 最 后 使 用 方法 display0 把 表单 显示 在 当前 的 浏览 器 上 。 

3.， 确认 数据 

验证 是 许多 Web 开发 语言 的 弱项 ， 但 是 在 PHP 中 使 用 HIML_QuickForm 可 以 很 好 地 解决 这 个 
问题 ， 使 PHP 无 论 在 客户 端 还 是 在 服务 器 端的 验证 都 非常 简单 。 它 通过 使 用 addRule0 方 法 对 提交 
的 数据 进行 相应 的 校 验 ， 之 后 ， 用 validate0 方 式 执行 验证 。 在 上 面 显示 了 没有 被 验证 的 表单 (因为 
在 这 个 时 候 处 理 表单 的 代码 能 够 被 访问 )。 如 果 要 实现 数据 格式 的 校 验 ， 一 切 都 很 简单 。 
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下 面 创建 一 个 案例 ,演示 使 用 HIML QuickFormm 执行 数据 的 校 验 。 这 个 例子 有 两 个 规则 , 一 个 
是 【姓名 】 文 本 框 必须 要 有 包含 的 内 容 ， 第 二 个 是 【姓名 】 文 本 框 中 只 能 包含 字母 并 且 不 能 是 符号 。 
打开 记事 本 ， 输 入 下 列 代码 : 


案例 12-3 

require once "HTML/QuickForm.php™"; 
$form = new HTML QuickForm('register', "post'") 7 
$form->addElement ('text', 'firstName', ' 姓 名 '); 
$form->addElement ('password', 'password' ，' 密 码 '); 
$form->addElement ('textarea', 'ta',' 详 细 信 息 '); 
$form->addElement ('submit','sb', ' 提 交 '); 
$form->addRule ('firstName'，' 该 文本 域 不 能 为 空 '，'required'); 
$form->addRule ('firstName'，“' 该 文本 域 只 能 包含 字母 ', 'lettersonly'); 
if ($form->validate()) { 

echo “验证 通过 ' ; 
} 
else { 

$form->display (); 
. 


大 

将 上 述 代码 保存 ,文件 名 为 Html0.php,， 并 将 该 文件 保存 在 C:\Web\apache\htdocs\PE 目录 下 。 以 
后 的 案例 如 果 没 有 特别 声明 都 是 保存 在 目录 下 。 打 开 正 浏览 器 ， 在 地 址 栏 中 输入 
http://localhost:8080/PE/Html0.php， 单 击 【 转 到 】 按 钮 ， 会 显示 如 图 12-21 所 示 的 窗口 。 

在 图 12-21 中 ， 如 果 在 【姓名 】 文 本 框 中 没有 相应 的 输入 内 容 或 者 输入 内 容 中 包含 非 字母 字符 ， 
单 击 【提交 】 按 钮 ， 会 显示 如 图 12-22 所 示 的 窗口 。 
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图 12-21 界面 显示 图 12-22 数据 校 验 


在 本 案例 中 , 有 两 个 方法 需要 注意 , 一 个 是 addRule0 方 法 , 该 方法 主要 是 执行 数据 格式 的 指定 ， 
如 果 没 有 满足 指定 的 格式 就 不 能 够 通过 验证 。addRule0 方 法 具有 三 个 参数 ， 分 别 表示 要 设 定格 式 的 
文本 元 素 、 提 示人 信息、 指定 格式 。 另 外 一 个 是 方法 validate0， 该 方法 表示 数据 校 验 通过 后 ， 自 动 执 
行 数据 的 处 理 。 

4. 处 理 表单 数据 

迄今 为 止 ， 已 经 简单 地 成 功 显示 创建 的 表单 并 进行 了 数据 的 校 验 ， 接 下 来 可 以 处 理 表单 中 的 数 
据 。HIML _ QuickFomm 提供 一 个 processO 函 数 ， 用 于 调用 另外 一 个 函数 并 审核 提交 值 。 其 使 用 示例 


如 下 所 示 : 


<? 
require once ("HTML/QuickForm.php"); 
$form = new HTML QuickForm('frmTest', ‘post'); 
$form->addElement ("header'， 'header'，' 请 登录 '); . 
$form->addElement ('text'， 'name'，“' 用 户 名 : '); 
$form->addElement ('password'， 'password'，“' 密 码 : '); 
$form->addElement ('submit'，“''，“' 提 交 '); 
if ($form->validate()) { 
$form->process('say hello'); 
} else { 


$form->display (); 第 
} 人 忆 
function say hello($data) { 章 
print ' 你 好 ，' . $data['name']; 局 
print "<BR> "7 Ss 
print "你 的 密码 是 ' . $data['password']; 刀 
1 
a 


在 上 述 代码 中 ， 需 要 注意 的 是 验证 方法 validate0 中 的 操作 。 在 此 方法 中 ， 首 先 调用 process() 方 
法 ， 而 process() 方 法 调用 另外 一 个 函数 say_hello0 执 行 表单 中 提交 的 数据 操作 。say_hello0 函 数 主要 
是 输出 表单 中 提交 的 数据 ， 其 中 data 表示 传递 的 表单 对 象 参数 。 

5. 格式 化 表单 

早期 的 HTML _QuickForm 版 本 在 改变 表单 布局 时 并 不 灵活 。 但是， 从 3.0 版 本 开始 有 众所周知 
的 Visitor 设计 样式 。 有 8 个 renderer 可 用 ， 而 且 有 以 下 这 些 模板 引擎 直接 支持 : Smarty、 
HTML Template_ Sigsma、HTMIL Template IT、HTMIL Template Flexy。 这 样 可 以 在 显示 PHP 页 面 
时 ， 制 定 表单 在 客户 端 显示 的 格式 。 其 使 用 示例 如 下 所 示 : 


<? 
require once "HTML/QuickForm.php"; 
$defaults = array("firstName"=>"Ian", 
"password"=>"abc", 
"ta"=>"Describe yourself here"); 
$form = new HTML QuickForm('register', "post')7 
$form->addElement ('text', 'firstName'， ' 姓 名 '); 
$form->addElement ('password', 'password', "密码 ') ; 
$form->addElement ('textarea'，'ta',，' 备 注 '); 
$form->addElement ('submit'，'sb',，' 提 交 '); 
$form->addRule ('firstName'，' 姓 名 不 能 为 空 '， 'required'); 
Srenderer =& $form->defaultRenderer (); 
$special element template=" 
<tr> 
<td align="right" valign="top"> 
<!-- BEGIN required --><span style="color: blue">*</span><!-— END required --> 
<font size="+22">{label}</font> 
</td> 


卫 
ud 
五 
十 
三 
op) 
[3 
| 
完 
全 
至 
习 
手 
册 


<td valign="top" align="left"> 
<!-- BEGIN error --><span style="color: magenta">{error}</span><br /><!-- END 
error -=> 
{element}</td> 
ES 
S$renderer->setElementTemplate ($special element template); 
if ($form->validate()) { 
echo ‘Success!'; 
else { 
S$form->setDefaults ($defaults); 
$form->display ();} 
?> 


在 上 述 代码 中 ， 使 用 语句 “S$renderer =& $form->defaultRenderer()” 创 建 模板 对 象 renderer， 然 
后 创建 一 个 标记 显示 变量 special_ element template， 指 定 表单 显示 的 样式 。 为 了 使 创建 的 格式 作用 
于 表单 ， 需 要 使 用 语句 “ $renderer->setElementTemplate($special element template)” 完 成， 其 中 
setElementTemplate() 方 法 表示 为 当前 的 模板 对 象 设置 指定 的 格式 。 


12.4.2 ”使 用 Calendar 创建 日 历 


Calendar 包 用 来 产生 Calendar 数据 结构 ， 能 够 自动 完成 很 多 与 时 间 有 关 的 任务 ， 它 并 不 显现 内 
容 ， 且 不 依赖 任何 数据 ， 所 以 可 以 被 应 用 到 众多 场合 。 同 时 它 提供 易 用 的 API 来 帮助 显示 一 个 
Calendar， 比 如 一 个 HTML Calendar， 而 且 可 以 将 数据 源 (data source) 和 Calendar 连接 在 一 起 。 该 包 
具有 多 种 用 途 ， 下 面 列 出 了 其 中 的 几 种 常见 的 用 途 : 

。 显示 指定 日 期 格式 输出 ( 如 (X)HTML、WML、SOAP、XML-RPC、 命 令 行 的 ASCII 等 )。 
生成 一 个 Calendar 如 同人 遍历 一 个 数据 库 查询 的 结果 集 。 
易于 生成 普通 的 非 弹 出 式 Calendar。 

所 有 日 期 可 以 很 容易 地 验证 其 能 效 性 。 
允许 使 用 分 层 的 概念 。 
可 以 通过 创建 装饰 器 来 附加 自己 需要 的 功能 。 
所 有 的 操作 返回 值 是 数值 ， 这 样 就 不 被 限制 使 用 英语 的 月 份 单词 ， 可 以 根据 自己 的 需要 从 返 
回 生成 对 应 的 数据 ， 可 以 通过 strfime() 和 setlocaleO 函 数 。 
。 Calendar 的 计算 引擎 被 抽象 化 ， 也 就 是 说 它 可 以 生成 各 种 各 样 的 Calendar 而 不 需要 修改 。 目 
前 的 引擎 是 基于 UNIX 时 间 蕉 。 

Calendar 是 一 个 大 包 ， 由 12 个 公共 类 组 成 ， 分 为 4 个 不 同 的 组 ， 分 别 为 日 期 类 、 表 格 日 期 类 、 
验证 类 、 装 饰 器 类 。 其 中 ， 日 期 类 用 于 管理 6 个 日 期 部 分 : 年 份 (Calendar Year )、 月 份 
(Calendar Month)、 日 (Calendar Day)、 小 时 (Calendar Hour)、 分 钟 (Calendar Minute ) 、 秒 
(Calendar Second)。 表 格 日 期 类 用 于 构建 基于 表格 的 月 历 和 周 历 ， 共 有 Calendar Month Weekdays、 
Calendar Month Weeks 和 Calendar Week 三 个 类 ， 这 些 类 分 别 用 于 构建 按 天 的 表格 日 历 、 按 星期 的 
表格 日 历 及 按 7 天 格式 的 日 历 。 验 证 类 主要 用 于 验证 日 期 的 格式 ， 有 Calendar Validator 和 


Calendar Validation_ Error 两 个 类 ， 分 别 用 于 验证 日 期 的 各 个 部 分 和 日 期 有 错误 时 ， 提 供 相 应 的 报错 
消息 。 装 饰 类 主要 用 于 扩展 其 他 子 类 。 

1. 安装 Calendar 

为 了 使 用 所 有 的 Calendar 包 的 所 有 特性 ， 
需要 把 该 包 的 依赖 包 Date 安装 ， 然 后 再 安装 
Calendar 包 。 其 安装 过 程 如 图 12-23 所 示 。 

2. 创建 月 历 

安装 完成 后 ， 就 可 以 使 用 PEAR 包 了 。 下 
面 创建 一 个 案例 ， 演 示 使 用 Calendar 包 。 打 开 
记事 本 ， 输 入 下 列 代码 : 图 12-23 安装 Calendar 


案例 12-4 
<?php 
require once 'Calendar/Month/Weekdays.php'; 
$month=new Calendar Month Weekdays (2007,4,0); 
$month->build(); 
echo "<table cellspacing='5'>\n"; 
echo "<tr><td class='monthname' colspan="7'> 七 月 2007</td></tr>"; 
echo "<tr><td>Su</td><td>Mo</td><td>Tu</td><td>We</td><td>Th</td><td>Fr</td> 
<td>sa</td>"; 
while ($day=$month->fetch()){ 
if($day->isFirst()){ 
echo "<tr>"; 
} 
if($day->isEmpty()){ 
echo "<td>gnbsp;</td>"; 
3 
elsef{ 
echo "<td>".$day->thisDay()."</td>"; 
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} 
if($day->isLast()){ 
echo "</tr>";}} 
echo "<table>"; 
?> 


将 上 述 代码 保存 ， 文 件 名 为 Calendarphp。 打 开 正 浏览 器 ， 在 地 址 栏 中 输入 http://localhost: 
8080/PE/Calendarphp， 单 击 【 转 到 】 按 钮 ， 会 显示 如 图 12-24 
所 示 的 窗口 。 

在 本 案例 中 , 语句 “$month=new Calendar Month Weekdays 
(2007.7.0)” 表 示 创 建 了 一 个 表示 月 的 表格 日 历 对 象 month， 该 
对 象 所 指向 的 日 期 是 2007 年 的 7 月 ， 日 历 应 该 从 星期 日 到 星期 
六 布局 ， 所 以 第 三 个 参数 设置 为 0， 它 表示 星期 日 的 数值 (1 表 
示 星 期 一 ，2 表示 星期 二 等 )。Build0 方 法 表示 生成 一 个 数组 ， 
包含 该 月 中 的 所 有 日 期 . 在 while 循环 中 使 用 fetch0 方 法 遍历 数 
组 中 的 每 一 个 日 期 数据 ， 并 输出 。isFirst0、isEmptyO、isLastO 


12-24 显示 日 期 
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方法 分 别 用 来 判断 是 否 为 第 一 个 数据 ， 数 据 是 否 为 空 ， 是 否 是 最 后 一 个 数据 等 。 

3. 验证 日 期 

在 Calendar 包 中 ， 可 以 使 用 方法 checkdate0 来 验证 日 期 。 但 是 ， 使 用 该 方法 时 需要 提供 三 个 相 
关 日 期 字段 参数 ， 分 别 为 月 份 、 日 和 年 。 如 果 只 想 验 证 其 中 的 一 个 日 期 字段 ， 例 如 年 日 期 字段 ， 可 
以 使 用 如 表 12-3 所 示 的 函数 。 


表 12-3 “日 期 函数 
名 称 说 了 明 
isValidO 执行 所 有 其 他 时 间 和 日 期 验证 方法 ， 验 证 日 期 和 时 间 


isValidDay0 | 确保 值 落 在 1~31 之 间 
isValidHour0 ”| 确保 值 落 在 0 一 23 之 间 
isValidMinute0 ”| 确保 值 落 在 0 一 59 之 间 
isValidMonthO ”| 确保 值 落 在 1 一 12 之 间 


isValidYear() 确保 在 UNIX 中 ,该 值 落 在 1902 一 2037 之 间 ; 在 Windows 中 ， 该 值 落 在 1970 一 2037 之 间 


12.4.3 ”使 用 AUTH_HTTP 认证 


前 面 章节 介绍 的 HITP 和 PHP 认证 ， 能 够 解决 认证 的 相关 问题 ， 但 是 这 样 做 ， 没 有 把 功能 的 实 
现 细节 隐藏 起 来 ， 容 易 暴 露 。 在 PEAR 中 存在 一 个 AUTH_HTTP 包 ， 能 够 很 好 地 封装 实现 的 细节 ， 
它 利 用 Apache 的 认证 机 制 和 提示 窗口 来 生成 相同 的 提示 ， 但 使 用 PHP 来 管理 认证 信息 。 
AUTH_ HTTP 封装 了 用 户 认证 的 很 多 杂乱 方面 ， 通 过 方便 的 接口 提供 了 需要 的 信息 和 特性 。 此 外 ， 
因为 它 继 承 自 AUTH 类 ， 所 以 AUTH_HTTP 还 提供 了 大 量 认证 存储 机 制 ， 包 括 DB 数据 库 抽象 包 、 
LDAP、 POP3、IMAP、RADIUS 和 SAMBA。 

1. 安装 AUTH_HTTP 

要 使 用 AUTH _HTTP 包 ， 需 要 进行 安装 。AUTH _HTTP 包 有 一 个 依赖 包 AUTH， 故 在 安装 时 ， 
应 首先 使 用 pear list 命令 查看 该 包 是 否 存在 ， 如 果 存 在 则 直接 安装 AUTH_HTTP 包 ， 和 否则 使 用 下 列 
命令 进行 安装 : 


pear install -o auth http 


打开 命令 提示 符 窗口 ， 输 入 上 面 的 命令 ， 会 在 窗口 中 显示 如 图 12-25 所 示 的 信息 。 

从 显示 信息 中 可 以 看 出 ， 这 里 只 安装 了 
AUTH HTTP 包 , 因为 AUTH 包 已 经 被 安装 了 。 

2. 根据 MySQL 数据 库 进行 认证 

AUTH_HTTP 可 以 方便 地 构建 基于 HTTP 
验证 的 程序 ， 而 且 可 以 自由 地 指定 用 户 数据 表 ， 
操作 简单 易 用 。 因 为 AUTH HTIP 是 AUTH 包 
的 子 类 ， 所 以 它 继承 了 AUTH 的 功能 。 而 AUTH 是 DB 包 的 子 类 ， 所 以 AUTH _ HTTP 可 以 利用 这 
个 流行 的 数据 库 抽象 层 ， 把 认证 信息 存储 在 数据 库 表 中 。AUTH HTTP 包 的 使 用 示例 如 下 所 示 : 

<?php 

require once("Auth/HTTP.php"); 

$options = array( 


图 12-25 安装 AUTH HTTP 包 


'dsn'=>"mysql://root:@localhost/test"，// 数 据 库 连接 字符 串 


'table'=>"test http", // 用 户 表 

'usernamecol'=>"name", // 用 户 名 字段 

'passwordcol'=>"passwd", // 用 户 密码 字段 有 

'cryptType'=>"md5", / /密码 加 密 方 式 . 

'db_fields'=>"#", oe 
. 


yp” 
$a = new Auth HTTP("DB", $options); 


$a->setRealm('yourrealm'); //realm name 

$a->setCancelText ('Error 401'); // 验 证 失败 时 的 提示 信息 

$a->start (); // starting the authentication process 
if ($a->getAuth () ) // 成 功 


{ 
echo "Hello $a->username welcome to my secret page"; 
echo $a->getAuthData('userid'); // 获得 其 他 字段 
echo $a->getAuthData('telephone'); 
echo $a->getAuthData('email'); 

}; 

> 


需要 注意 的 是 options， 这 个 数组 要 传递 到 AUTH_ HTTP 构造 函数 ， 并 声明 为 一 个 数据 类 型 。 
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12.4.4 使 用 HTTP_Upload 上 传 


实现 文件 的 上 传 有 多 种 方法 ， 其 中 利用 PEAR 包 中 的 HTTP_Upload 就 是 一 种 。HTTP_Upload 
实现 上 传 文件 , 能 够 很 好 地 封装 实现 的 细节 , 并 通过 一 个 便利 的 接口 提供 给 用 户 需要 的 信息 和 特性 。 

1. 安装 HTTP_Upload 

在 使 用 HTTP_Upload 之 前 ， 需 要 安装 该 PEAR 包 。 其 安装 命令 如 下 所 示 : 


pear install HTTP Upload 


在 命令 提示 符 窗口 ， 输 入 此 命令 后 ， 会 
显示 如 图 12-26 所 示 的 信息 。 

2. 获取 上 传 文件 信息 

使 用 HTTP_Upload 获取 上 传 的 文件 信 
息 非常 容易 ， 只 是 调用 了 几 个 方法 而 已 。 其 
使 用 示例 如 下 所 示 : 

<form action="" enctype="multipart/form-data" method="post"> 

文件 名 :<br/> <input type="text" name="name" value=""/><br/> 


上 传 文件 :<br/><input type="file" name="classnotes" value=""/><br/> 
<p><input type="submit" name="submit" value=" 上 传 "> 


图 12-26 安装 HTTP_Upload 包 


</form> 


<?php 
require ("HTTP/Upload.php"); 
$upload=new HTTP Upload(); 
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S$Sfile=Supload->getEFiles("classnotes")7 
$props=$file->getProp(); 
print r($props); 

Rx 


在 上 述 代码 中 ， 第 一 部 分 代码 主要 实现 获取 上 传 文件 的 显示 页 面 。 属 性 标记 “action=""” 表 示 
将 得 到 的 文件 信息 传递 给 本 页 面 。 在 第 二 部 分 代码 中 ， 实 现 上 传 文件 的 功能 。 这 里 使 用 语句 
“Tequire("HTTP/Upload.php")” 加 载 需要 使 用 的 PEAR 包 ， 并 创建 了 一 个 上 传 对 象 upload， 调 用 该 对 
象 的 getFiles0 方 法 获取 上 传 的 文件 ， 其 参数 是 上 面 表单 中 传递 过 来 的 。 然 后 调用 getProp0 方 法 获取 
上 传 文件 的 信息 ， 其 文件 信息 是 以 数组 的 形式 存储 起 来 的 。 

3. 指定 上 传 位 置 

在 上 面 代码 中 ， 只 是 实现 了 获取 上 传 文件 的 一 些 信息 ， 如 文件 大 小 、 名 称 等 ， 但 是 了 解 文件 的 
上 传 属性 还 是 远 远 不 够 的 ， 还 需要 设 定 上 传 文件 的 具体 位 置 。 下 面 创 建 一 个 案例 ， 演 示 上 传 文件 ， 
并 将 文件 放置 到 指定 的 位 置 。 打 开 记事 本 ， 输 入 下 列 代码 : 


案例 12-5 

<form action="Httpl.php" enctype="multipart/form-data" method="post"> 
上 传 文件 :<br/><input type="file" name="classnotes" value=""/><br/> 
<p><input type="submit" name="submit" value=" 上 传 "> 

</form> 


将 上 述 代码 保存 ， 文 件 名 为 Http0.php。 该 文件 主要 实现 上 传 文件 的 显示 页 面 。 打 开 记事 本 ， 创 
建 处 理 上 传 的 PHP 脚本 程序 ， 其 代码 如 下 所 示 : 


案例 12-5 
<?php 
require ("HTTP/Upload.php"); 
$upload=new HTTP_ Upload(); 
$file=$upload->getFiles ("classnotes"); 
if ($file->isValid()){ 
$file->moveTo("C:/web"); 
echo "文件 上 传 成 功 "; 
} 
else 


echo $file->errorMsg(); 
} 


2 

将 上 述 代 码 保存 ， 文 件 名 为 Httpl.php。 打 开 正 浏览 器 ， 在 地 址 栏 中 输入 http://localhost:8080/ 
PE/Http0.php， 单 击 【 转 到 】 按 钮 ， 会 显示 如 图 12-27 所 示 的 窗口 ， 在 窗口 中 选择 好 上 传 的 文件 ， 单 
击 【 上 传 】 按 钮 ， 会 显示 如 图 12-28 所 示 的 窗口 。 

在 本 案例 中 ， 使 用 isValid0 方 法 判断 要 上 传 的 文件 是 否 存 在 问题 ， 如 果 该 文件 没有 问题 则 使 
方法 moveTo0， 指 定 文件 在 服务 器 上 所 放置 的 位 置 ， 本 案例 中 文件 放置 的 位 置 为 “C:vweb”。 最 后 
一 行使 用 了 方法 errorMsg0， 这 个 方法 会 跟踪 很 多 潜在 的 错误 ， 如 上 传 目录 不 存在 、 缺 少 写 入 权限 、 
复制 错误 或 文件 大 小 超出 最 大 的 上 传 限制 等 问题 ， 默 认 情况 下 ， 这 些 提 示 信 息 为 英文 。 


驴 httpz//localhostz8080/PE/Httpl-php ---- 基因 | 属 | 
文件 四 “ 编 得 四“ 坦 看 中” 收 辣 内 工具 四 帮助。 局 


寺 让 加) 国 http-//1ocslhost-8080/PE/Httpl php > Ea 
文件 上 传 成 功 : 
局 二 地 Intranet oe 
图 12-27 显示 页 面 图 12-28 成 功 页 面 
4. 上 传 多 个 文件 
HTTP_Upload 包 不 但 可 以 一 次 上 传 一 个 文件 ， 还 可 以 一 次 上 传 多 个 文件 。 如 果 要 实现 这 样 的 操 
作 ,， 只 需 使 用 该 类 的 实例 化 对 象 ， 对 每 个 上 传 文件 调用 getFiles0 方 法 就 可 以 了 。 下 面 创建 一 个 案例 ， 第 
演示 一 次 性 上 传 多 个 文件 。 打 开 记事 本 ， 输 入 下 列 代码 : 12 
案例 12-6 了 
<form action="Http3.php" enctype="multipart/form-data" method="post"> 
上 传 文件 :<br/><input type="file" name="classnotesl" value=""/><br/> 号 


上 传 文件 :<br/><input type="file" name="classnotes2" value=""/><br/> 
上 传 文件 :<br/><input type="file" name="classnotes3" value=""/><br/> 
<p><input type="submit" name="submit" value=" 上 传 "> 

</form> 


将 上 述 代 码 保存 ， 文 件 名 为 Http2.php。 该 文件 主要 实现 上 传 文件 的 客户 端 显 示 。 打 开 记 事 本 ， 
创建 处 理 上 传 的 PHP 脚本 程序 ， 其 代码 如 下 所 示 : 


案例 12-6 

<?php 
require ("HTTP/Upload.php"); 
$upload=new HTTP_ Upload(); 
$filel=$upload->getFiles ("classnotes1"); 
$file2=$upload->getFiles ("classnotes2"); 
$file3=$upload->getFiles("classnotes3"); 
if ($filel->isValid()){ 

$filel->moveTo("C:/web"); 


echo "第 一 个 文件 上 传 成 功 "; 


时 
else 
{echo $filel->errorMsg();} 
if ($file2->isValid()){ 
$file2->moveTo("C:/web"); 
echo "第 二 个 文件 上 传 成 功 "; 
中 
else 
{echo $file2->errorMsg();} 
if ($file3->isVvalid()){ 
$file3->moveTo("C:/web"); 


echo "第 三 个 文件 上 传 成 功 "; 
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else 
{echo $file3->errorMsg();} 
2 


将 上 述 代码 保存 ， 文 件 名 为 Http3.php。 打 开 正 浏览 器 ， 在 地 址 栏 中 输入 http://localhost:8080/ 


PE/Http2.php， 单 击 【 转 到 】 按 钮 ， 会 显示 如 图 12-29 所 示 的 窗口 ， 在 File 域 中 选择 要 上 传 的 文件 ， 
单 击 【 上 传 】 按 钮 ， 会 显示 如 图 12-30 所 示 的 窗口 。 


到 http://1oralhost:8080/PE/Http3.php - 1..。 恬 | 加 | 区 


ET TI TE 大 
第 一 个 文件 上 传 成 功 第 二 个 文件 上 传 成 功 
第 三 个 文件 上 传 成 功 


下地 Intranet 
图 12-29 上 传 多 个 文件 图 12-30 上 传 成 功 


在 本 案 中 ， 代 码 和 上 一 个 案例 的 差别 不 是 太 大 ， 但 是 本 案例 处 理 的 是 多 个 文件 上 传 。 如 果 上 传 
文件 太 多 ， 也 可 以 使 用 循环 来 处 理 。 
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很 多 网 站 都 希望 当 用 户 浏览 网 页 时 ， 能 够 跟踪 用 户 的 行踪 ， 但 是 HTTP 协议 没有 记忆 性 ， 当 月 
户 从 一 个 页 面 转 到 另外 一 个 页 面 时 ，HTTP 协议 不 能 记 住 该 用 户 的 身份 ， 因 为 用 户 对 当前 Web 页 面 
的 请 求 被 认为 是 唯一 的 和 独立 的 连接 ， 与 之 前 的 连接 没有 关系 。 

要 记 住 用 户 的 信息 及 用 户 的 当前 状态 ， 一 个 可 行 的 解决 办 法 就 是 在 客户 端 保存 一 些 数据 ， 这 样 
当 用 户 再 次 访问 时 ， 可 以 请 求 从 用 户 本 地 文件 中 取出 相关 信息 ， 这 就 是 Cookie 和 Session 会话 ) 
技术 的 基本 原理 。 


FY 内 容 摘 要 1pstsct 


掌握 Cookie 的 基本 操作 及 Cookie 的 删除 
理解 Cookie 的 工作 原理 及 其 用 途 
掌握 如 何 控 制 Cookie 的 有 效 性 
理解 什么 是 能 力 监视 器 
掌握 会 话 的 基本 用 法 

理解 如 何 配置 PHP 的 会 话 
掌握 如 何 传输 会 话 ID 及 获得 会 话 ID 
掌握 如 何 使 用 会 话 存储 数据 

理解 会 话 存储 的 工作 原理 
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13.1 Cookie 概述 


Cookie 是 一 种 Web 服务 器 通过 浏览 器 在 用 户 的 硬盘 上 存储 信息 的 手段 。 将 Cookie 用 于 Intemet 
是 为 了 克服 HTTP 无 状态 记录 的 缺陷 ， 所 以 可 以 说 Cookie 就 是 由 Web 服务 器 保存 到 用 户 端 本 地 计 
算 机 上 的 数据 ， 通 常 以 文件 的 方式 保存 在 硬盘 上 。 


13.1.1 基本 操作 


Cookie 在 许多 方面 都 是 很 有 用 的 ， 本 节 将 介绍 Cookie 的 基本 操作 ， 也 就 是 如 何 创建 和 访问 
Cookie。 

1. 创建 Cookie 

要 想 创建 Cookie， 需 要 使 用 setcookie0 函 数 ， 它 的 具体 使 用 格式 如 下 所 示 : 
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bool setcookie ( string name [，string value [，int expire [，string path [，string 
domain [, int secure]]]]]) 


上 述 使 用 格式 定义 一 个 随 其 他 HTTP 头 文件 一 起 发 送 的 Cookie。Cookie 必须 在 其 他 所 有 HTTP 
头 文件 发 出 前 发 送 ， 也 就 是 说 ， 它 应 该 在 文件 的 最 前 面 。 

该 函数 的 所 有 参数 中 除了 name 参数 外 都 是 可 选 的 。 如 果 只 有 一 个 name 参数 ， 则 表示 删除 客户 
端 浏览 器 中 名 字 为 name 的 Cookie。 也 可 以 通过 设置 为 空 字符 串 来 跳 过 一 个 参数 , 但 expire 和 secure 
的 值 为 整数 ， 不 能 用 空 字符 串 ， 只 能 用 0 来 代 蔡 。expire 参数 是 一 个 由 time0 或 者 mktime0 函 数 返回 
的 UNIX 格式 的 函数 。secure 表示 Cookie 只 在 建立 HTTP 连接 时 才 发 送 。 

使 用 timeO 函 数 可 以 返回 从 1970 年 1 月 1 日 以 来 按 秒 计算 的 经 过 时 间 。 可 以 添加 一 个 偏 移 值 ， 
它 指定 应 该 可 以 访问 Cookie 的 时 间 。 例如 , 现 假设 创建 一 个 名 为 username 的 Cookie, 它 包 括 值 “ 宋 
岩 岩 ”。 这 个 Cookie 在 创建 后 的 1h (3600s) 内 是 可 用 的 。 具体 语 句 如 下 所 示 : 


setcookie ("username"," 宋 岩 岩 ", time ()+3600) 7 
除了 使 用 time0 函 数 外 ， 用 户 还 可 以 使 用 mktime0 函 数 指定 过 期 时 间 。 该 函数 的 具体 格式 如 下 
所 示 : 


int mktime ( [int hour [, int minute [, int second [, int month [, int day [, int 
year [, int is dst]]]]]]]) 


上 述 使 用 格式 根据 给 出 的 参数 返回 UNIX 时 间 戳 。 其 中 ， 参 数 可 以 从 右 向 左 省 略 ， 任 何 省 略 的 
参数 都 会 被 设置 成 本 地 日 期 和 时 间 的 当前 值 。is_dst 在 夏令 时 可 以 被 设 为 1， 如 果 不 是 则 设 为 0， 或 
者 不 知道 是 否 为 夏令 时 间 则 设 为 -1 默认 值 )。 如 果 不 知道 ，PHP 会 尝试 自己 判断 ， 这 可 能 会 产生 
未 预期 (但 不 是 错误 ) 的 结果 。 例 如 ， 下 面 的 语句 将 创建 一 个 Cookie， 它 将 在 2008 年 1 月 1 号 的 
第 一 秒 钟 过 期 : 


setcookie ("username"," 宋 岩 岩 ", mktime (0,0,1,1,1,2008)); 


2. 访问 Cookie 

如 果 用 户 已 经 创建 了 一 个 Cookie， 则 自动 将 这 个 Cookie 的 值 作为 一 个 PHP 变量 使 用 ， 这 个 变 
量具 有 与 这 个 Cookie 相同 的 名 称 。 前 面 创建 了 一 个 名 为 username 的 Cookie, 并 为 它 赋值 “ 宋 岩 岩 ”， 
这 样 与 Web 站 点 相关 联 的 每 一 个 PHP 脚本 都 可 以 使 用 这 个 名 称 - 值 对 。 如 下 所 示 为 访问 Cookie 的 具 
体 示例 : 


<pre> 
<?php 
setcookie ("usernamel"," 宋 岩 岩 ",time() + 3600); 
setcookie ("username2"," 龙 建华 ",time() + 3600) 
// 以 下 语句 用 于 输出 某 一 个 Cookie 
echo "The Usernamel is ".$HTTP COOKIE VRRS ["usernamel"] ."\n"7 
echo "The Usernamel is ".$ COOKIE["usernamel"]."\n"; 
// 以 下 语句 用 于 输出 所 有 的 cookie 
print r($ COOKRIE); 
汪汪 
</pre> 


成 功 执行 上 述 语句 ， 将 输出 如 下 所 示 的 信息 : 


The Usernamel is 宋 岩 岩 
The Usernamel is 宋 岩 岩 
Array 
f 
[usernamel] => 宋 岩 岩 
[username2] => 龙 建华 
h 


这 里 需要 注意 的 是 ，Cookie 值 是 由 浏览 器 作为 HTTP 头 的 一 部 分 发 送 的 。 因 此 ， 在 向 浏览 器 发 
送 任何 输出 之 前 ， 必 须 设置 Cookie 值 。 即 使 发 送 一 个 单独 的 空格 也 将 无 法 设置 Cookie 值 。 为 了 避 
免 出 现 问题 ， 一 定 要 将 设置 Cookie 值 的 PHP 脚本 放 在 文件 的 顶部， 并 且 前 面 没有 空白 字符 。 此 外 ， 
还 应 该 在 向 浏览 器 发 送 输 出 的 echo 语句 或 男 一 个 PHP 语句 执行 之 前 设置 Cookie 值 。 


13.1.2 Cookie 如 何 工作 


Cookie 经 常用 于 在 Web 浏览 器 中 保存 应 用 程序 的 状态 。 因 为 与 用 Get 或 Post 发 送 的 数据 在 一 
起 ， 因 此 Cookie 利用 浏览 器 发 出 的 HTTP 请 求 来 发 送 。Cookie 是 保存 在 Web 浏览 器 中 的 一 条 指定 
信息 。 浏 览 器 可 以 使 用 JavaScript 创建 Cookie， 但 是 ，Cookie 通常 在 Set-Cookie 首部 字段 中 作为 一 
个 HTTP 响应 从 Web 服务 器 发 送 到 客户 端 。 考 虑 一 个 HTTP 响应 的 例子 : 


HTTP/1.0 200 

Content-Length: 1276 

Content-Type: text/html 

Date: Tue, 06 Nov 2007 04:12:49 GMT 
Expires: Tue, 06 Nov 2007 04:12:59 GMT 
Server: yanwebs/3.1.6 

Set-Cookie: animal=egg-laying-mammal 
<html>...</html> 


接收 该 响应 的 Web 浏览 器 记 住 Cookie， 并 将 它 作为 首部 字段 Cookie 包含 在 发 送 到 同一 Web 服 
务 器 的 后 续 HITP 请 求 中 。 例 如 ， 如 果 一 个 浏览 器 接收 到 刚才 显示 的 这 个 响应 ， 则 后 续 的 请 求 具有 
以 下 格式 : 

GET /jack/bill.php HTTP/1.0 

Connection: Keep-Alive 

Cookie: animal=egg-laying-mammal 

Host: www.itzcn.com 

Referer: http://www.itzcn.com/ 


Cookie 是 通过 HTTP 头 来 设置 的 ， 服 务 器 端 使 用 set-Cookie header 来 创建 一 个 Cookie， 设置 
Cookie 的 值 和 一 些 属 性 ， 当 客户 端 浏览 器 在 向 服务 器 发 送 请 求 时, 使 用 Cookie header 可 以 把 创建 的 
Cookie 发 送 到 服务 器 。 

set-Cookie HTTP 头 包含 5 个 参数 ， 如 表 13-1 所 示 。 
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表 13-1 ”选项 说 明 
使 用 格式 


Cookie-name=Cookie-value 


参 数 


Cookie-name 


说 ”了 明 
Cookie-name 是 要 设置 的 Cookie 变量 的 名 称 ， 而 
Cookie-value 就 是 要 设置 的 值 


expires | expires=expiration-date 该 参数 设置 Cookie 保存 的 时 间 期 限 

path path=Cookie-path 该 参数 用 来 指定 Cookie 将 被 发 送 到 服务 器 的 哪 一 个 
目录 路 径 下 ， 与 域名 一 起 工作 ， 确 定 何 时 发 送 该 
Cookie 变量 
对 浏览 器 发 送 Cookie 进行 限制 , 该 Cookie 只 会 被 发 


送 到 指定 的 服务 器 上 ， 绝 对 不 会 被 发 送 给 其 他 网 站 
该 参数 表示 Cookie 通过 HTTPS 协议 加 密 后 传输 


domain | domain=server-domain 


secure 


下 面 来 看 一 个 示例 ， 如 下 所 示 : 


Set-Cookie:username=JackLong;path=/;domain=Wwww.itzcn.com; 
Expires=Sunday, 30-June-2007 01:24:06 GMT;secure 


上 面 的 HTTP 头 自动 在 浏览 器 的 Cookie 文件 中 创建 一 条 记录 ,将 名 为 “usemame” 的 变量 的 值 
设置 为 “JackLong”， 过 期 时 间 是 2007 年 6 月 30 日 1 时 24 分 6 秒 ,该 Cookie 只 能 够 被 发 送 到 
www.itzen.com 根 目 录 下 ， 使 用 HTTPS 协议 加 密 后 发 送 。 


13.1.3 ”控制 Cookie 的 有 效 性 


自从 Cookie 技术 出 现 以 来 ， 就 成 为 网 络 用 户 和 Web 开发 人 员 争 论 的 焦点 。 一 方面 它 可 以 跟踪 
用 户 在 网 站 中 的 访问 情况 ， 收 集 一些 统 计 信息 ， 还 能 方便 地 保存 用 户 信息 ， 比 如 账号 、 密 码 以 及 用 
户 在 Web 站 点 的 订购 信息 ， 这 是 它 有 利 的 一 面 ， 但 是 ， 另 一 方面 ， 一 些 网 络 用 户 甚至 网 络 专家 对 
Cookie 技术 不 满 ， 认 为 Cookie 的 使 用 侵犯 了 用 户 的 隐私 ， 而 且 它 会 对 用 户 计算 机 的 安全 性 造成 威 
胁 ， 因 为 他 们 担心 Cookie 中 记录 的 一 些 个 人 信息 会 被 某 些 别有用心 的 人 收集 ， 对 用 户 造成 损害 。 

虽然 对 于 Cookie 有 着 争议 ,不 过 从 目前 的 发 展 情况 来 看 , 绝 大 多 数 的 网 络 用 户 都 能 接受 Cookie， 
因为 这 项 技术 实在 太 重 要 了 。 在 进行 网 站 开发 时 ， 使 用 Cookie 技术 可 以 为 网 站 增色 不 少 ， 所 以 怎 
样 控制 Cookie 的 有 效 性 是 非常 重要 的 ， 合 理 地 控制 Cookie 的 有 效 性 将 有 效 地 保护 网 络 用 户 信 息 的 
安全 。 

下 面 的 示例 就 是 模拟 用 户 的 登录 情况 , 用 户 根据 自己 的 实际 情况 选择 不 同 的 选项 来 设置 Cookie 
的 有 效 期 ， 从 而 最 大 限度 地 保护 用 户 信息 的 安全 。login.php 文件 用 于 实现 上 述 功 能 ， 它 的 具体 代码 
如 下 所 示 : 

案例 13-1 

<html> 

<head> 

<meta http-equiv="Content-Type" content="text/html; charset=gb2312" /> 

<title> 控 制 Cookie 的 有 效 性 </title> 

</head> 

<body> 

<center> 

<table width="500" border="1"> 


savcookie.php 页 面 用 于 将 用 户 的 信息 存储 在 Cookie 中 , 而 它 的 有 效 期 是 根据 用 户 的 选择 进行 设 
定 的 。 它 的 具体 代码 如 下 所 示 : 
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else 
{ 
if ($usernameCookie == "") 
{ 
echo "Cookie 数据 已 经 顺利 存储 <hr><p>"; 
echo "目前 \$usernameCookie 是 : [".S$username; 
echo "] 有 效 期 是 : " .$savtime." 秒 <p>"; 
} 
else 
{ 
echo "Cookie 数据 已 经 顺利 存储 <hr><p>"; 
echo "目前 \$usernameCookie 是 : [".$usernameCookie; 
echo "] 有 效 期 是 : " .$savtime." 秒 <p>"; 
} 
} 
echo "<br><a href='1ogin.php'> 重 新 输入 </a>"; 
2 
</center> 
<body> 
</body> 
</html> 


把 上 述 文件 login.php 和 savcookie.php 存储 在 Apache 目录 下 的 htdocs\13 子 目 录 下 ,打开 正 浏 
览 器 ， 在 地 址 栏 中 输入 http:Wlocalhost/13/login php， 按 Enter 键 会 显示 如 图 13-1 所 示 的 页 面 。 


EE 
文件 9) 入 可) 查看 W) 收藏 w) 工具) 于 用 00 


二 全 器) 糊 http://1ocalhost/13/1egin phe 同 加 和 到 5” 
A 


请 给 入 你 的 姓名 ， 


请 锁 入 你 的 密码 : 


Cookie 有 效 期 [10 分钟 国 
确证 委 交 


ET 


图 13-1 loginphp 页 面 


在 该 页 面 中 输入 用 户 姓 名 及 密码 ， 并 选择 相应 的 Cookie 有 效 期 ， 然 后 单 击 【 确 定 提交 】 按 钮 ， 
会 显示 如 图 13-2 所 示 的 页 面 。 
马 适 过 传 入 的 时 间 谈 署 Cookic 的 有 效 其 


文件 FF) ” 辊 辑 正 ) 查看 W) 收 章 人 ) 工具 上 ) 
EECORECTF 


Cooki ce 数据 已 经 顺利 存 钳 


目前 $usernameCookie 是 :，[ 龙 建华 ] 有 效 期 是 ，504800 秒 


重新 往 入 


13-2 ”savcookie.php 页 面 


在 案例 13-1 中 , 通过 手动 方式 设置 Cookie 的 有 效 期 来 控制 Cookie 的 有 效 性 , 这 样 Cookie 就 会 
在 用 户 希 望 的 时 间 失 效 ， 从 而 能 有 效 地 保护 用 户 信息 的 安全 。 如 果 读 者 留心 会 发 现 这 种 方式 的 使 用 
是 非常 普遍 的 。 


13.1.4 删除 Cookie 


因为 Cookie 有 一 个 过 期 时 间 ， 所 以 在 创建 它 之 后 的 某 个 时 刻 ， 它 将 被 自动 删除 ， 但 是 用 户 也 可 
以 立即 删除 一 个 Cookie。 要 想 实现 这 一 点 ， 将 Cookie 的 过 期 时 间 设置 为 过 去 的 一 个 时 间 即 可 。 例 
如 ， 要 删除 前 面 创建 的 username， 可 以 使 用 如 下 所 示 的 方法 : 


setcookie ("username","", time()-3600); 


上 述 语句 将 Cookie 的 过 期 时 间 设置 为 1h (3600s) 之 前 。 注意， 这 个 Cookie 的 值 被 指定 为 一 个 
空 字符 串 。 由 于 这 个 Cookie 不 再 可 用 ， 因 此 它 的 值 也 就 不 重要 了 。 
另外 ， 还 有 一 个 比较 简单 的 方法 ， 具 体 如 下 所 示 : 


setcookie ("username"); 


下 面 对 前 面 访 问 Cookie 的 示例 进行 修改 ， 来 展示 删除 Cookie 是 如 何 工作 的 ， 有 具体 语句 如 下 
所 示 : 

<pre> 

<?php 
setcookie ("usernamel", " 宋 岩 岩 "， time() + 3600); 
setcookie ("username2"," 龙 建华 ",time() + 3600); 
echo "The Usernamel is ".$ COOKIE["usernamel"]."\n"; 
echo "The Username2 is ".$ COOKIE["username2"]."\n"; 
setcookie ("usernamel"); 
setcookie ("username2","",time() - 3600); 
print r($ COOKIE); 

?> 

</pre> 


成 功 执行 上 述 语句 ， 将 输出 如 下 所 示 的 信息 : 


The Usernamel is 宋 岩 岩 
The Username2 is 龙 建华 
Array 
( 
[usernamel] => 
[username2] => 


) 
13.1.5 ”Cookie 数组 


因为 在 PHP 中 ， 一 个 Web 站 点 只 能 在 用 户 系 统 上 存储 20 个 Cookie， 所 以 能 够 在 一 个 单独 的 
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Cookie 中 存储 多 个 值 是 非常 有 用 的 。 要 想 做 到 这 一 点 ， 可 以 将 值 放 在 一 个 数组 中 ， 还 可 以 使 用 
serialize0 函 数 将 数组 的 元 素 序列 化 到 一 个 字符 串 中 ， 然后 使 用 unserialize0 函 数 恢复 数组 的 值 。 下 面 
的 示例 显示 了 如 何 创建 和 访问 包含 多 个 值 的 Cookie， 具 体 语 句 如 下 所 示 : 


下 面 来 看 一 个 Cookie 数组 较为 简单 的 示例 ， 具 体 语 句 如 下 所 示 : 


这 里 需要 注意 ， 一 个 浏览 器 同一 时 刻 能 够 创建 的 Cookie 数目 最 多 为 300 个 ， 同 一 个 Web 服务 
器 对 一 个 客户 端 最 多 只 能 发 送 20 个 Cookie， 每 个 Cookie 的 大 小 不 能 够 超过 4KB， 如 果 想 存储 超过 
4KB 的 数据 ， 则 必须 将 数据 存储 在 一 个 服务 器 端 数据 库 或 者 Cookie 之 外 的 其 他 位 置 。 


13.1.6 ”把 什么 放 到 Cookie 中 


由 于 客户 端的 每 一 项 数据 各 自 独立 ， 在 需要 读 取 时 再 分 别 取出 。Cookie 就 是 将 数据 存放 在 客户 
端的 技术 ， 它 的 使 用 过 程 如 图 13-3 所 示 。 
如 果 把 数据 存放 在 服务 器 ， 那 么 管理 起 来 就 比较 方便 ， 有 些 数据 一 定 要 存放 在 服务 器 ， 例 如 ， 


用 户 的 账号 名 称 、 密 码 等 ,但 是 有 些 数据 
并 不 是 那么 重要 ， 例 如， 用 户 昵称 、 用 户 
访问 本 站 的 次 数 等 , 如 果 把 这 些 信息 也 都 
存放 在 服务 器 中 ， 不 仅 数据 存储 的 开销 
大 ， 而 且 搜索 庞大 的 数据 ， 需 要 花费 不 少 
的 时 间 ， 将 对 服务 器 的 性 能 产生 影响 。 若 ”Web 服 务 器 湛 用 户 端 
能 将 这 些 个 人 数据 存放 在 用 户 计算 机 中 ， 图 13.3 Cookie 的 使 用 过 各 
将 大 大 减轻 服务 器 的 负担 。 

大 部 分 的 Web 程序 开发 者 ， 都 会 利用 Cookie 来 存储 不 太 重 要 的 用 户 个 人 数据 ， 而 将 重要 的 认 
证 数据 ， 例 如 ， 交 易 数量 、 客 户 ID、 密 码 等 存储 在 服务 器 中 。 


13.2 会 话 


会 话 (Session) 是 为 特定 用 户 识 别 和 管理 状态 (会 话 变量 ) 的 一 种 方式 。 当 用 户 发 送 一 个 HITP 
请 求 时 ， 中 间 层 必须 在 用 户 的 会 话 环 境 中 处 理 当 前 请 求 。 当 一 个 会 话 开始 后 ， 将 会 给 客户 端 一 个 与 
服务 器 的 后 续 请 求 包含 在 一 起 的 会 话 ID(SID)。 服 务 器 使 用 会 话 ID 在 处 理 请 求 之 前 定位 相应 的 会 话 。 
本 节 将 介绍 有 关 会 话 的 内 容 。 


13.2.1 基本 用 法 


取出 Cookie 中 的 数据 


数据 处 理 后 显示 在 页 面 上 


本 节 将 介绍 会 话 是 如 何 开始 的 以 及 如 何 创建 和 删除 会 话 变 量 。 

1. 开始 会 话 

由 于 HITP 不 能 记忆 用 户 过 去 和 将 来 的 环境 ， 因 此 ， 需 要 对 每 次 请 求 显 式 地 启动 和 恢复 会 话 。 
用 户 可 以 使 用 session_start0 函 数 来 完成 这 两 项 值 的 任务 。session_start0 函 数 的 具体 使 用 格式 如 下 
所 示 : 

bool session start ( void ) 


session_start0 函 数 创建 一 个 新 会 话 或 继续 当前 会 话 ， 这 取决 于 是 否 拥 有 SID。 开 始 会 话 时 ， 只 
需 如 下 调用 session_start0 函 数 。 具 体 如 下 所 示 : 


session start(); 


这 里 需要 注意 的 是 ， 无 论 结果 如 何 ，session_start0 函 数 都 会 报告 成 功 的 结果 。 因 此 ， 使 用 任何 
异常 处 理 都 不 起 作用 。 

2. 创建 会 话 变量 

用 户 可 以 通过 session_register0 函 数 来 创建 会 话 变 量 . 如 果 一 个 会 话 没有 初始 化 , session_registerO 
函数 将 调用 session_start0 函 数 打 开会 话 文件 。 可 以 通过 session_register0 函 数 调 用 来 注册 (添加 到 会 
话 中 ) 变量 。 具 体 示例 如 下 所 示 : 
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// 创 建 会 话 变量 Myweb 
session register("MyWeb") 7 
SMYWeb = "http://www.itying.net"; 


这 里 应 该 注意 ， 是 将 变量 的 名 称 而 不 是 变量 本 身 传递 给 session_register0 函 数 。 一 旦 创建 ,会话 
变量 就 变 得 持久 并 对 初始 化 会 话 的 脚本 可 用 。PHP 追踪 会 话 变量 的 值 并 将 它们 的 值 保存 到 会 话 文件 
中 ， 所 以 必须 在 脚本 结束 之 前 显 式 地 保存 会 话 变 量 。 上 述 语句 中 变量 MyWeb 与 它 的 值 被 自动 保存 
在 会 话 存储 器 中 。 

现在 首选 的 方法 是 : 只 需 像 其 他 任何 变量 一 样 设置 会 话 变量 即 可 ， 但 是 需要 在 _SESSION 超级 
全 局 上 下 文中 引用 这 些 变 量 。 例 如 ， 假 设 希望 创建 名 为 usemame 的 会 话 变量 ， 具 体 语句 如 下 所 示 : 


<?php 

session start(); 

$_SESSION['username'] = " 宋 岩 岩 "; 

echo "The UserName is:".$ SESSION['username']; 
?> 


成 功 执 行 上 述 语 句 ， 将 输出 如 下 所 示 的 信息 : 
The Username is: 宋 岩 岩 


3 删除 会 话 变 量 
用 户 可 以 使 用 unsetO 函 数 删除 会 话 变量 。 下 面 修改 创建 会 话 变 量 的 语句 ， 有 具体 示例 如 下 所 示 : 


<?php 
session start(); 
$_SESSION['username'] = " 宋 岩 岩 "; 
echo "The UserName is:".$ SESSION['username'] ."<br>"7 
unset($ SESSION['username'])7 
echo "Now The UserName is:".$ SESSION['username']."<br>"; 
?> 


成 功 执行 上 述 语句 ， 将 输出 如 下 所 示 的 信息 : 


The UserName is: 宋 岩 岩 
Now The UserName is: 


另外 ,也 可 以 利用 session_unregister0 函 数 调用 来 删除 会 话 变 量 , 这 里 作为 参数 的 是 变量 的 名 称 
而 不 是 变量 本 身 。 注 销 的 变量 对 初始 化 该 会 话 的 其 他 脚本 不 再 可 用 。 但 是 ， 该 变量 对 紧 跟 在 
session_UnregisterO 函 数 调用 之 后 的 脚本 的 其 余部 分 仍然 可 用 。 


13.2.2 配置 PHP 的 会 话 


配置 PHP 的 会 话 是 通过 设置 配置 指令 来 完成 的 ， 这 些 指令 用 于 确定 PHP 会 话 处 理 功能 的 行为 。 
下 面 将 对 这 些 指令 进行 详细 的 介绍 ， 如 表 13-2 所 示 。 


指 令 


表 13-2 PHP 的 会 话 指令 
说 明 


session.save_ handler (string) 


该 指令 用 于 确定 如 何 存储 会 话 信息 ， 它 可 以 通过 以 下 4 种 方法 之 一 进行 存 
储 : 文件 (files)、 共 享 内 存 (mm)、SQLite 数据 库 〈sqlite) 和 用 户 自 定义 
函数 (user)， 它 的 作用 域 是 PHP_INI ALL; 默认 值 是 fles。 其 中 ， 人 多 es 选项 
会 话 存 储 文件 的 数量 可 能 会 达到 几 千 个 ， 甚 至 在 一 段 时 间 后 会 达到 几 十 万 
个 ; 共享 内 存 选项 速度 最 快 , 但 却 最 不 可 靠 , 因为 数据 存储 在 RAM 中 ; sqlite 
选项 利用 了 新 的 SQLite 扩展 ,通过 这 个 轻 量 级 数据 库 透 明 地 管理 会 话 信息 ; 
而 用 户 自 定义 选项 虽然 配置 复杂 ， 但 也 是 最 灵活 、 功 能 最 强大 的 一 个 选项 ， 
因为 可 以 创建 定制 处 理 函 数 ， 在 开发 人 员 所 需 的 任何 媒体 上 存储 信息 


session.save_path (string) 


session.name (string) 


session.auto_start (boolean) 


session.serialize_ handler (string) 


Session.gc_probability (integer) 


session.gc_maxlifetime (integer) 


该 指令 与 session.save_handler 指令 结合 使 用 , 如 果 session.save_handler 设置 
为 files 存储 选项 ， 则 该 指令 必须 指向 存储 目录 。 这 里 需要 注意 的 是 ， 该 指 
令 不 能 设置 为 位 于 服务 器 文档 根 中 的 某 个 目录 ， 否 则 可 以 轻松 地 通过 浏览 
器 危害 这 些 信息 的 安全 。 另 外 ， 所 指向 的 存储 目录 必须 是 服务 器 守护 进程 
可 写 的 。 

通常 情况 下 ， 出 于 效率 方面 的 考虑 ， 可 以 使 用 N:/path 来 定义 session.save_ 
path， 其 中 N 是 一 个 整数 ， 表 示 可 以 存储 会 话 数据 的 N 层 深度 子 目录 的 数 
量 。 该 指令 的 作用 域 是 PHP INI ALL; 默认 值 是 /mp 

该 指令 用 于 确定 Cookie 名 ， 它 的 作用 域 是 PHP INL ALL; 默认 值 是 
PHPSESSID。 这 里 可 以 把 默认 值 改 为 更 适合 于 应 用 程序 的 名 字 ， 或 者 可 以 
通过 session nameO 函 数 在 需要 时 进行 修改 

该 指令 用 于 确定 是 否 自动 启动 会 话 ， 它 的 作用 域 是 PHP_INI_ALL; 默认 值 
是 0。 如 果 将 该 指令 设置 为 1， 则 自动 启动 会 话 ， 通 过 函数 session_start0 可 
以 显 式 启动 会 话 。 如 果 用 户 要 在 网 站 中 一 直 使 用 会 话 ， 可 以 考虑 设 此 指令 
为 1， 否则 可 以 设置 为 0， 以 在 必要 时 调用 session_start0 函 数 启动 会 话 。 
这 里 需要 注意 ， 启 动 该 指令 将 无 法 在 会 话 中 存储 对 象 ， 因 为 类 定义 要 在 会 
话 开始 之 前 加 载 才能 重新 创建 对 象 ， 但 session.auto_start0 函 数 会 禁止 这 一 
点 ， 所 以 如 果 要 在 会 话 中 管理 对 象 ， 就 要 禁用 该 指令 

该 指令 定义 了 串 行 化 和 逆 串 行 化 数据 所 用 的 回调 处 理 器 ， 它 的 作用 域 是 
PHP_INI_ALL; 默认 值 是 php。 默 认 情 况 下 ， 这 由 称 为 PHP 的 内 部 处 理 器 
来 处 理 。 PHP 还 支持 另 一 个 串 行 化 处 理 器 , 即 Web 开发 数据 交换 (WDDX)， 
在 提供 WDDX 支持 的 条 件 下 编译 PHP 就 可 以 使 用 这 个 串 行 化 处 理 器 

该 指令 用 于 定义 所 有 概率 的 分 子 部 分 ， 而 该 分 子 部 分 用 于 计算 调用 垃圾 回 
收 过 程 的 频率 。 其 中 ， 分 母 由 下 面 介 绍 的 指令 session.gc_divisor 指定 

该 指令 用 于 确定 有 效 会 话 的 持续 时 间 ， 以 秒 为 单位 ， 它 的 作用 域 是 
PHP INI ALL; 默认 值 是 1440。 当 到 达 此 限制 时 (默认 24 min, 即 1440 s)， 
会 话 信息 将 被 销毁 ， 系 统 资源 得 以 重新 分 配 


SessionLIeferer check (string) 


session.entropy_file (string) 


该 指令 用 于 指定 一 个 用 于 验证 每 位 用 户 的 子 串 ， 如 果 没有 包含 此 子 串 ，SID 
将 失效 。 如 果 使 用 URL 重 写 来 传播 会 话 ID, 这 样 很 多 人 都 能 通过 复制 和 散 
布 URL 来 查看 某 个 会 话 状态 ， 使 用 该 指令 可 以 减少 这 种 可 能 性 

该 指令 用 来 指向 一 个 额外 的 信息 源 〈entropy source)， 并 把 这 个 信息 源 集成 
到 SID 生成 过 程 中 。 在 Windows 系统 下 ,安装 Cygwin (http://www.cygwin. 
com) 会 提供 类 似 于 random 或 urandom 的 功能 。 而 在 UNIX 系统 下 ， 这 个 
来 源 通 常 是 /dev/random 或 /dev/urandom 


session.entropy_length (integer) 


该 指令 确定 从 session.entropy_file 所 指定 的 文件 中 读 取 字 节 数 。 如 果 
session.entropy_file 为 空 ， 则 此 指令 被 忽略 ， 并 使 用 标准 的 SID 生成 模式 
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指 令 


续 表 
说 了 明 


session.use_cookies (boolean) 


该 指令 设置 为 1, 表示 使 用 Cookie 进行 SID 传播 ; 设置 为 0 将 使 用 URL 重 
写 。 它 的 作用 域 是 PHP_INI ALL; 默认 值 是 1。 如 果 要 在 多 次 访问 网 站 之 
间 维 护 用 户 会 话 ， 就 应 当 使 用 Cookie， 这 样 处 理 器 可 以 重新 得 到 SID， 继 
续 原来 所 保存 的 会 话 。 如 果 用 户 数据 只 用 于 一 次 访问 , 那么 就 可 以 使 用 URL 
重 写 


session.use_only_cookies (boolean) 


session.cookie lifetime (integer) 


session.cookie path (string) 


session.cookie_domain (string) 


session.cookie_secure (boolean) 


session.cache limiter (string) 


该 指令 用 来 确定 只 使 用 Cookie 来 维护 SID, 而 忽略 通过 URL 传递 SID 的 尝 
试 。 它 的 作用 域 是 PHP_INI ALL; 默认 值 是 0。 设 置 此 指令 为 1 时 ，PHP 
只 使 用 Cookie， 设 置 为 0 则 同时 使 用 Cookie 和 URL 重 写 

该 指令 用 于 确定 会 话 Cookie 的 有 效 期 ， 以 秒 为 单位 。 它 的 作用 域 是 
PHP INI ALL; 默认 值 是 0。 如 果 设 置 该 指令 为 0， 则 Cookie 将 一 直 存活 
到 浏览 器 重新 启动 。 例 如 ， 要 将 Cookie 存活 lh， 就 将 该 指令 设置 为 3600 
该 指令 用 来 确定 Cookie 在 哪个 路 径 中 是 有 效 的 。 对 于 这 个 有 效 路 径 下 的 所 
有 子 目录 ， 此 Cookie 同样 有 效 。 例 如 ， 如 果 该 指令 设置 为 /， 则 Cookie 对 
整个 网 站 都 有 效 。 如 果 设 置 为 /bbs， 则 只 有 在 http://www.itzcn.conybbs 路 径 
中 调用 Cookie 才 有 效 

该 指令 确定 Cookie 在 哪个 域 中 有 效 , 它 的 作用 域 是 PHP _INI ALL; 默认 值 
是 空 。 该 指令 是 必需 的 ， 因 为 它 能 防止 其 他 域 读 取 用 户 的 Cookie。 该 指令 
的 设置 如 下 所 示 : 


session.cookie _ domain = www.itzcn.com 


如 果 要 在 网 站 子 域 中 使 用 会 话 , 假如 , 现在 有 bbs.itzcn.com、books.itzcn.com 
和 download .itzcn.com 子 域 ， 该 指令 可 以 如 下 设置 : 


session.cookie domian = .itzcn.com 


该 参数 设置 Cookie 的 安全 标记 ， 这 样 可 以 防止 浏览 器 通过 不 加 密 的 连接 发 
送 会 话 Cookie。 当 该 参数 设置 为 On 时 , 浏览 器 通过 一 个 使 用 SSL (安全 套 
接 字 层 ) 保护 的 网 络 连 接 发 送 会 话 Cookie。 它 的 作用 域 是 PHP_INI_ALL; 
默认 值 是 Off。 默 认 值 Of 允许 浏览 器 通过 加 密 和 不 加 密 的 服务 发 送 会 话 
Cookie 

该 指令 用 来 确定 会 话 页 面 是 否 被 缓存 及 如 何 缓存 。 它 可 以 被 设置 为 以 下 5 
个 慎之 二 i 

nono 该 设置 禁止 随 启用 会 话 的 页 面 传输 任何 缓存 控制 首部 。 

nocache 该 设置 确保 对 于 每 个 请 求 ， 在 可 能 提供 缓存 的 版 本 之 前 ， 先 
将 请 求 发 送 到 最 初 的 服务 器 。 

private ”该 设置 指定 缓存 的 文档 是 私有 的 ,该 文档 只 用 于 最 初 的 用 户 ， 
不 能 与 其 他 用 户 共享 。 

private no_expire 该 设置 是 private 的 变 体 , 它 保证 不 会 向 浏览 器 发 送 
任何 文档 的 过 期 日 期 。 

public 该 设置 认为 所 有 文档 都 是 可 缓存 的 ,最初 的 文档 请 求 可 能 需要 
该 指令 的 作用 域 是 PHP INI ALL; 默认 值 是 nocache 


Session.cache_expire (integer) 


该 指令 用 来 确定 在 创建 新 页 面 之 前 ， 缓 存 的 会 话 页 面 可 用 的 时 间 〈 以 秒 为 
单位 )。 它 的 作用 域 是 PHP_INI ALL; 默认 值 是 180。 如 果 session.cache 
limiter0 函 数 设置 为 nocache， 则 该 指令 将 被 忽略 


指 令 


session.use trans_sid (boolean) 


说 了 明 
该 指令 被 启用 后 ， 可 以 消除 使 用 URL 重 写 过 程 中 发 生 错 误 的 可 能 性 。 它 的 
作用 域 是 PHP INI SYSTEM | PHP_INI PERDIR; 它 的 默认 值 是 0。 如 果 禁 
用 session use_cookies， 为 了 确保 传播 ID， 用 户 的 SID 必须 增加 到 URL。 
用 户 可 以 手动 将 变量 SSID 追加 到 每 个 URL 的 后 面 ,也 可 以 通过 启用 指令 自 
动 处 理 
该 指令 的 作用 域 是 PHP _ INI ALL; 默认 值 是 a=hrefarea=href frame=src、 
input=src,form=fakeentry。 启 用 session.use_trans_sid 时 ， 在 将 文档 发 送 给 客 
户 端 之 前 ，SID 会 自动 追加 到 所 请 求 文档 中 的 HTML 标记 后 面 。 但 是 ， 与 
超 链接 或 表单 标记 不 同 ， 许 多 标记 在 启动 服务 器 请 求 时 不 起 作用 ， 所 以 可 
以 使 用 该 指令 告诉 服务 器 SID 应 当 追 加 到 哪些 标记 后 面 。 该 指令 的 使 用 类 
似 于 如 下 所 示 : 


Url _ rewriter.tags (string) 


url rewriter.tags a=href,frame=src,form=,fieldset= 


13.2.3 ”如 何 传输 会 话 1D 


PHP 脚本 第 一 次 调用 session_start0 函 数 时 , 将 产生 一 个 会 话 ID, 并 且 默 认 情 况 下 会 在 响应 中 包 
含 一 个 Set-Cookie 首部 字段 。 响 应 以 名 称 PHPSESSID 和 会 话 ID (SID) 的 值 在 浏览 器 中 设置 一 个 
会 话 Cookie。PHP 会 话 管理 自动 包含 该 Cookie， 不 需要 调用 setcookie0 或 header0 函 数 。 

会 话 ID 是 32 个 十 六 进 制 数 字 的 随机 字符 串 ， 比 如 fcc17f071bca9bf7f85ca281094390b4。 与 其 他 
Cookie 一 样 ， 会 话 ID 的 值 在 HITP_ COOKIE VARS 关联 数组 和 PHPSESSID 变量 中 对 PHP 脚本 
可 用 。 

当 开 始 一 个 新 会 话 时 ，PHP 创建 一 个 会 话 文件 。 在 默认 配置 下 ， 会 话 文件 使 用 会 话 ID 加 前 级 
sess_ 作 为 文件 名 写 到 /tmp 目录 中 。 与 示例 会 话 ID 关联 的 文件 名 是 /tmp/sess_fec17f071bca9bf 
7f85ca281094390b4， 其 中 会 话 ID 主要 使 用 以 下 两 种 方法 进行 传输 。 

1. 使 用 Cookie 

用 户 在 访问 网 站 时 ， 会 在 必要 时 获取 SID， 并 在 页 面 中 使 用 与 SID 相关 的 各 项 数据 。 这 样 ， 即 
使 会 话 结束 ，Cookie 也 能 在 客户 端 保存 ， 所 以 可 以 在 后 续 会 话 中 读 取 ， 即 使 用 户 长 时 间 没 有 进行 任 
何 操作 ， 也 能 持久 地 保存 信息 。 但 这 里 需要 注意 的 是 ， 由 于 用 户 可 以 通过 浏览 器 控制 是 否 接受 
Cookie， 所 以 在 进行 程序 设计 时 要 考虑 到 这 方面 的 问题 。 

2. 使 用 URL 

用 户 可 以 通过 在 每 个 请 求 页 面 的 本 地 URL 上 增加 会 话 ID 来 实现 它 的 传输 。 无 论 用 户 何 时 单 击 
这 些 本 地 链接 ， 都 会 导致 会 话 ID 的 自动 传播 ， 这 里 所 采用 的 方法 被 称 为 URL 重 写 。 这 样 即 使 客户 
端 禁用 Cookie， 也 不 会 影响 网 站 会 话 处 理 功能 的 正常 使 用 。 但 是 ， 这 种 方法 也 存在 一 定 的 问题 : 首 
先 ，URL 重 写 在 会 话 之 间 不 能 保证 持久 性 ， 因 为 一 旦 用 户 离开 网 站 ， 向 URL 自动 追加 会 话 ID 的 过 
程 就 无 法 再 持续 。 其 次 , 无 法 阻止 用 户 将 URL 复制 到 电子 邮件 并 发 送 给 另 一 位 用 户 , 并 且 只 要 会 话 
没有 超时 ， 会 话 就 会 在 接收 者 的 计算 机 上 持续 。 如 果 两 个 用 户 同步 使 用 相同 的 会 话 ， 或 者 链接 的 接 
收 者 无 意 中 看 到 了 会 话 的 秘密 数据 ， 都 有 可 能 发 生 严重 灾难 ， 所 以 出 于 安全 方面 的 考虑 ， 建 议 使 用 
基于 Cookie 的 方法 ， 但 具体 如 何 实现 可 以 根据 自己 的 项 目 进行 决定 。 
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如 果 调 用 session_start0 函 数 ， 并 且 请 求 包含 PHPSESSID Cookie，PHP 将 试图 查找 会 话 文件 并 
初始 化 关联 的 会 话 变量 。 但 是 ， 如 果 识 别 的 会 话 文件 未 能 找到 ，session_start0 函 数 将 创建 一 个 空 的 
会 话 文件 。 

不 要 将 PHPSESSID 的 存在 用 作 新 会 话 的 标识 ， 或 者 用 作 访 问 会 话 ID 的 方法 。 在 第 一 次 调用 肢 
本 和 创建 会 话 时 ，PHPSESSID Cookie 可 能 还 没有 设置 ， 只 有 后 续 的 请 求 能 确保 包含 PHPSESSID 
Cookie。PHP 中 提供 了 session_ id0 函 数 ， 用 于 返回 初始 化 会 话 的 会 话 卫 。 关 于 它 的 使 用 将 在 后 面 进 
行 详细 介绍 。 


13.2.4 ”使 用 会 话 存 储 数据 


使 用 会 话 存储 数据 就 是 使 用 创建 的 会 话 变 量 存储 数据 ， 所 以 会 话 变 量 的 类 型 决定 了 会 话 要 存储 
的 数据 。 会 话 变量 可 以 是 布尔 型 、 整 型 、 双 精度 型 、 字 符 串 型 、 对 象 型 ， 或 者 是 这 些 变量 类 型 的 数 
组 。 使 用 对 象 会 话 变量 时 必须 小 心 ， 因 为 当初 始 化 一 个 已 有 会 话 时 ，PHP 需要 访问 已 注册 的 对 象 的 
类 定义 。 如 果 对 象 被 保存 为 会 话 变量 ， 用 户 应 该 将 这 些 对 象 的 类 定义 保存 在 初始 化 会 话 的 所 有 脚本 
中 ， 而 不 管 脚本 是 否 使 用 类 。 

PHP 通过 将 值 串 行 化 〈serialize) 而 将 会 话 变量 保存 在 会 话 文件 中 。 变 量 的 串 行 化 表示 将 名 称 、 
类 型 和 值 包含 为 一 个 适合 于 写 到 文件 的 字符 流 。 该 示例 类 似 于 如 下 所 示 : 


count|i:6;start|i:986084651 


在 下 面 使 用 会 话 存 储 数据 的 示例 中 , 创建 了 两 个 变量 : 一 个 是 整 型 的 count, 每 调用 一 次 该 变量 
就 自动 增加 1; 另 一 个 是 start， 当 会 话 第 一 次 初始 化 时 ， 就 设置 为 从 库 函 数 ttme0 返 回 的 当前 时 间 。 
示例 中 通过 测试 会 话 变量 count 是 否 已 注册 来 确定 是 否 创建 了 一 个 新 的 会 话 。 如 果 变量 count 还 没有 
注册 ， 它 将 自动 增加 1。 所 以 ，count 用 于 显示 脚本 被 调用 了 多 少 次 ， 而 time0-$start 用 于 显示 会 话 
持续 了 多 少 秒 。 具 体 代码 如 下 所 示 : 


案例 13-2 
<?php 
// 初 始 化 一 个 会 话 
// 这 个 调用 要 么 创建 一 个 新 的 会 话 ， 要 么 重建 一 个 已 有 的 会 话 
session start(); 
// 如 果 这 是 一 个 新 的 会 话 ， 那 么 将 不 注册 变量 count 
if(!session is registered("count")) 
4 
session register("count"); 
$ SESSION['start'] = time(); 


$count = 0; 
} 
else 
' 

Scount++7 


} 


$sessionId = session id(); 
LE 


<html> 
<head> 
<title>Session 的 使 用 </title> . 
</head> 
<body> 
<p> 
The Session Identification Number is :<?=$sessionId?> 
<br>count=<?=$count?>. 
<br>start=<?=$_SESSION['start']?>. 


<p>This Session Has Lasted: 第 
<2php BS 
$dur=time ()-$start; 有 
echo "$dur"; OO 
?> 己 
和 
</body> 和 
</html> 会 
话 


把 上 述 代码 存储 在 Apache 目录 下 的 htdocs\13 
子 目录 下 ， 文 件 名 为 sessionsave. php， 打开 正 浏 
览 器 ， 在 地 址 栏 中 输入 http://localhost/13/session- 
save.php， 按 Enter 键 会 显示 sessionsave.php 页 面 ，|™ Hae Losted: L18371se64W 


重复 刷新 几 次 页 面 后 如 图 13-4 所 示 。 


13.2.5 “页面 缓存 


n Nunber is :17a48154babd9addgap6063539145246 


图 13-4 sessionsave.php 页 面 


对 于 一 个 日 访问 量 达 到 百 万 级 的 网 站 来 说 ， 速 度 成 为 制约 网 站 发 展 的 一 个 瓶颈 。 如 何在 有 限 的 
资源 下 提高 网 站 访问 速度 ? 除了 优化 内 容 发 布 系统 的 应 用 本 身 外 ， 如 果 能 把 不 需要 实时 更 新 的 动态 
页 面 的 输出 结果 转化 为 静态 网 页 来 发 布 ， 速 度 上 的 提升 效果 将 是 显著 的 。 另 一 方面 ， 如 果 静 态 网 页 
的 内 容 能 被 缓存 在 内 存 里 ， 那 么 访问 速度 将 会 进一步 提高 。 

1. 页 面 缓存 的 相关 概念 

这 里 先 介绍 一 下 缓存 ， 它 可 以 分 成 两 类 : 静态 缓存 和 动态 缓存 。 虽 然 都 采用 了 缓存 技术 ， 但 它 
们 的 实现 方式 却 不 同 。 下 面 将 对 它们 进行 分 别 介绍 。 

(1) 静态 缓存 在 新 内 容 发 布 的 同时 立刻 生成 相应 内 容 的 静态 页 面 。 例 如 ，IT 在 中 国 的 管理 员 通 
过 后 台 内 容 管理 界面 于 2007 年 7 月 3 日 录入 一 篇 文章 后 ， 将 立刻 生成 http://www.itzcn.com/article/ 
2007-07-03.html 这 个 静态 页 面 ， 并 同步 更 新 相关 索引 页 上 的 链接 。 

(2) 动态 缓存 在 新 内 容 发 布 以 后 ， 并 不 生成 相应 的 静态 页 面 ， 当 对 相应 的 内 容 发 出 请 求 时 ， 如 
果 前 台 缓 存 服务 器 找 不 到 相应 的 缓存 ， 就 向 后 台 内 容 管理 服务 器 发 出 请 求 ， 后 台 系 统 将 会 生成 相应 
内 容 的 静态 页 面 。 所 以 在 用 户 第 一 次 访问 页 面 时 可 能 会 慢 一 点 ， 但 是 以 后 就 是 直接 访问 缓存 了 。 

2. 页 面 缓存 的 定义 及 工作 原理 

页 面 缓存 是 指 整个 页 面 缓存 于 内 存 ， 页 面 缓存 是 动态 缓存 中 最 常用 的 一 种 缓存 技术 ， 以 后 对 该 
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页 的 请 求 将 由 缓存 输出 ， 而 不 必 创建 该 页 的 代码 。 

页 面 缓存 的 工作 原理 是 ， 首 先 判断 是 否 存在 该 页 的 缓存 文件 。 如 果 不 存在 则 提取 该 页 的 HTML 
代码 生成 缓存 文件 ， 然 后 输出 。 如 果 存在 则 判断 缓存 文件 的 修改 时 间 是 否 过 期 ， 如 果 没 有 过 期 则 输 
出 缓存 文件 。 如 果 缓 存 文件 缓存 时 间 过 期 ， 则 重新 提取 HTML 代码 生成 缓存 文件 进行 输出 。 该 实现 
代码 如 下 所 示 : 


实现 页 面 缓存 所 调用 的 类 ， 具 体 代码 如 下 所 示 : 


date ("Y"))); 
//tt 为 秒 数 

$tt=$filemtime now-$filemtime; 
if($tt>$this-> CACHE TIME) 


// 如 果 缓 存 文 件 的 最 后 修改 时 间 大 于 缓存 时 间 ， 则 更 新 缓存 
Sthis->Cache create($cash file,Sthis-> URL,Sthis-> QUERY); 
return $cash file; 


} 


else 


{ 
// 如 果 存 在 缓存 并 且 是 最 新 的 
return $cash file; 
} 
} 
else 
{ 
// 如 果 两 者 都 不 符合 ， 则 建立 缓存 文件 
: “// 这 部 分 代码 读者 根据 实际 情况 编辑 
} 
1 


?> 


13.2.6 ”破坏 会 话 


尽管 用 户 可 以 配置 PHP 的 会 话 处 理 指令 , 根据 过 期 时 间或 概率 自动 破坏 会 话 , 但 有 时 手动 破坏 
会 话 也 是 很 有 必要 的 。 例 如 ,可 能 希望 让 用 户 在 离开 网 站 时 注销 登录 , 或 当 用 户 单 击 适当 的 链接 时 ， 
可 以 从 内 存 中 消除 当前 会 话 中 的 所 有 变量 ， 甚 至 从 存储 机 制 中 完全 消除 整个 会 话 ， 这 可 分 别 通过 
session_unset() 和 session_destroy0 函 数 完成 。 

1. session_unset() 

该 函数 的 具体 使 用 格式 如 下 所 示 : 


void session unset () 

该 函数 用 于 清除 存储 在 当前 会 话 中 的 所 有 变量 ， 它 能 有 效 地 将 会 话 重 置 为 创建 时 的 状态 (没有 
注册 任何 会 话 变 量 的 状态 )。 这 里 需要 注意 的 是 , 这 并 非 从 存储 机 制 中 完全 删除 会 话 。 如 果 希 望 完全 
删除 会 话 ， 需 要 使 用 函数 seesion_destroy0。 

2. session_destroy() 

该 函数 的 具体 使 用 格式 如 下 所 示 : 

bool session destroy () 

该 函数 用 于 从 存储 机 制 中 完全 删除 会 话 ， 使 当前 会 话 失效 ， 但 该 函数 不 会 销毁 用 户 浏览 器 中 的 
任何 Cookie。 如 果 用 户 不 想 在 会 话 之 后 使 用 Cookie, 只 需要 在 php.ini 文件 中 将 session.cookie lifttime 
设置 为 0。 关 于 这 个 函数 的 具体 使 用 类 似 于 如 下 所 示 : 

<?php 
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// 只 在 请 求 设置 了 一 个 PHPSESSID 时 
// 才 试 图 结束 会 话 
if(isset($PHPSESSID)) { 
Smessage = "<p>End of session ($PHPSESSID)."™; 
session start(); 
session destroy(); 
} else { 
Smessage = "<p>There was no session to destroy!"; 
. 
?> 
<html> 
<head><title>Sessions</title></head> 
<body> 
<?=$message?> 
</body> 
</html> 


13.2.7 会 话 存 储 如 何 工 作 


这 里 要 介绍 的 会 话 存储 如 何 工作 也 就 是 Session 的 工作 原理 。 首 先 ，PHP 为 创建 Session 的 用 户 
产生 一 个 独一无二 的 字符 串 , 用 来 标志 该 用 户 的 会 话 。 通常 情况 下 , 将 这 个 字符 串 称 作 会 话 ID(SID)。 
然后 使 用 “sess” 十 会 话 IJD 为 文件 名 《假设 一 个 会 话 ID 为 fcc17f071bca9bf7f85ca28109439711， 那 
么 文件 名 为 sess_fec17f071bca9bf7f85ca28109439711) 在 服务 器 的 文件 系统 中 建立 一 个 文件 , 在 文件 
中 保存 用 户 在 Session 中 所 定义 的 全 局 变量 的 变量 名 和 值 。 然 后 再 将 会 话 ID 作为 一 个 名 为 
PHPSession 的 Cookie 保存 在 用 户 端的 文件 系统 中 。 

然后 ， 当 该 用 户 再 次 连接 服务 器 访问 一 个 PHP 脚本 时 ，PHP 从 用 户 发 来 的 PHPSession 这 个 
Cookie 中 得 到 用 户 所 在 Session 的 会 话 ID, 并 根据 会 话 ID 从 服务 器 的 文件 系统 中 查找 相应 的 Session 
信息 文件 。 最 后 从 这 个 文件 中 读 取 该 用 户 在 上 次 连接 时 所 设置 的 全 局 变量 的 值 。 

因此 , 可 以 看 到 Session 的 工作 原理 和 用 户 身份 认证 的 工作 原理 是 一 样 的 。 所 不 同 的 只 是 Session 
将 信息 保存 在 服务 器 的 文件 系统 中 ， 而 我 们 将 信息 保存 在 数据 库 中 。 使 用 Session 的 好 处 是 数据 的 
保存 和 获取 是 由 PHP 自动 完成 的 ， 如 果 直 接 使 用 Cookie 就 需要 自己 动手 进行 数据 的 保存 和 获取 。 

Session 利用 Cookie 的 身份 标志 功能 ， 将 用 户 在 浏览 网 站 时 需要 保存 的 信息 保存 在 服务 器 上 。 
这 样 Session 既 克服 了 HTTP 协议 的 缺陷 ， 又 防止 了 信息 的 泄漏 ， 而 且 方便 了 程序 设计 者 的 使 用 ， 
是 一 个 非常 好 的 解决 方案 。 

在 PHP 中 每 一 个 Session 都 通过 调用 session_startO 函 数 开 始 , 这 个 函数 检查 一 个 Session 是 否 
在 ， 如 果 不 存在 则 创建 一 个 新 的 。 然 后 用 session_register0 函 数 来 创建 一 个 会 话 变量 , 它 将 生存 于 整 
个 Session 中 。 

用 于 演示 Session 工作 原理 的 标准 例子 之 一 就 是 页 面 计数 器 ， 这 是 一 个 简单 的 基于 Session 的 计 
数 器 ， 在 用 户 第 一 次 访问 一 个 Web 页 面 时 初始 化 一 个 变量 ,每 一 次 当 用 户 重新 装 入 这 个 页 面 时 ， 该 
变量 的 值 会 自动 增 1。 具 体 代码 如 下 所 示 : 


<?php 
// 初 始 化 一 个 session 


Session Start () 7 


// 注 册 一 个 session 变量 
//$_SESSION['counter']; 
session register('counter'); 


// 增 加 计数 器 

$counter=$ SESSION['counter']+1; 

$ SESSION['counter']=$counter; 

echo "你 已 经 访问 了 本 页 面 : ” .$_SESSION['counter'] ." 次 "; 

?> 

当 用 户 每 一 次 重 装 这 个 页 面 时 ， 计 数 器 的 值 都 在 原来 的 基础 上 增加 1。 多 次 重 装 这 个 页 面 后 将 
显示 类 似 于 如 下 所 示 的 信息 : 


你 已 经 访问 了 本 页 面 : 26 次 


13.3 ”会 话 的 安全 性 


会 话 能 为 黑客 冯 入 系统 提供 方法 ， 也 能 为 黑客 打开 系统 攻击 的 大 门 提供 途径 ， 黑 客 可 能 以 合法 
用 户 的 身份 登录 到 应 用 程序 后 进行 破坏 性 操作 ， 所 以 关于 会 话 安全 性 的 问题 是 不 可 忽视 的 。 本 章 将 
介绍 获得 会 话 了 D 的 函数 session_id0 的 使 用 ， 以 及 如 何 限 制 泄密 的 会 话 世 造成 的 损害 。 


13.3.1 获得 会 话 1D 


会 话 ID(SID) 将 所 有 的 会 话 数 据 绑 定 到 某 个 特定 用 户 上 。 虽 然 PHP 能 够 自动 创建 和 传播 SID， 
但 有 时 也 希望 由 手动 获取 和 设置 这 些 SID。 用 户 可 以 通过 session_ id0 函 数 来 完成 这 两 项 任务 。 它 的 
具体 使 用 格式 如 下 所 示 : 

string session id ( [string id]) 

该 函数 可 以 设置 和 获得 SID。 如 果 没 有 参数 ， 函 数 session id0 返 回 当前 SID。 如 果 包 括 可 选 参 
数 i4， 则 当前 SD 将 被 该 值 蔡 换 。 具 体 示 例如 下 所 示 : 

<?php 


session start(); 
echo "The Session Identification Number is ".session id(); 
?> 


成 功 执行 上 述 语句 ， 将 输出 类 似 于 如 下 所 示 的 信息 : 

The Session Identification Number is qkuplqsnvpel299i696chcrc00 
如 果 用 户 想 设置 SID 的 值 ， 可 以 把 上 述 语句 进行 如 下 所 示 的 修改 : 

<?php 


session start (); 
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session id('Jacklong@www.itzcn.com'); 
echo "The Session Identification Number is ".session id(); 
> 


成 功 执行 上 述 语句 ， 将 输出 类 似 于 如 下 所 示 的 信息 : 


The Session Identification Number is Jacklong@www.itzcn.com 


13.3.2 ”限制 泄密 的 会 话 1D 造成 的 损害 


由 于 使 用 Cookie 和 URL 保持 与 某 个 用 户 相关 联 有 可 能 导致 会 话 DD 的 泄露 , 所 以 用 户 使 用 自 定 


卫 义 的 会 话 处 理 器 来 提供 最 大 程度 的 灵活 性 。 为 了 更 好 地 理解 后 面 的 程序 代码 ， 这 里 先 来 了 解 一 些 相 
到 关 的 函数 及 其 定义 。 其 详细 信息 如 表 13-3 所 示 。 
三 表 13-3 ”函数 及 其 定义 
名 画 数 说 阴 
到 bob_open($session save_path,$session name) 该 函数 初始 化 在 会 话 过 程 中 可 能 用 到 的 所 有 元 素 。 它 的 两 个 
全 输入 参数 session_save_path 和 session name 都 是 在 php.ini 
学 文件 中 找到 的 配置 指令 。 用 户 可 以 用 PHP 的 get_cfg_var0) 
习 函数 获取 这 些 配置 
让 bob_close0 该 函数 与 一 般 处 理 器 的 操作 类 似 ， 用 于 关闭 session_open() 
函数 初始 化 时 打开 的 资源 。 它 没有 参数 ， 也 不 能 破坏 会 话 
bob_read($sessionID) 该 函数 从 存储 介质 中 读 取 会 话 数据 ， 其 中 参数 sessionID 用 
于 标识 特定 客户 的 数据 的 会 话 ID 
bob_write($sessionID.$value) 该 函数 将 会 话 数据 写 入 存储 介质 ， 其 中 参数 sessionID 是 变 
量 名 ，value 是 会 话 数据 
bob_gc($lifetime) 该 函数 有 效 地 删除 所 有 到 期 的 会 话 ， 其 中 参数 lifetime 值 为 
php.ini 文件 中 的 会 话 配置 指令 session_gc_maxlifetime 
bob_destroy($session) 该 函数 用 于 销毁 会 话 和 所 有 相关 的 会 话 变量 ， 其 中 参数 


session 表示 当前 打开 会 话 的 SID 
bool session_set_save_handler ( callback $open. | 该 函数 为 系统 函数 , 用 于 将 用 户 定义 的 处 理 器 函数 加 入 PHP 
callback $close，callback Sread，callback $write, | 会话 处 理 逻 辑 中 。 其 中 的 参数 就 是 前 面 定义 的 处 理 器 函数 ， 
callback $destroy. callback $gc ) 它们 必须 以 正确 的 顺序 传 给 session_set_save_handler0 函 数 : 
打开 、 关 闭 、 读 取 、 写 入 、 销 毁 和 垃圾 回收 ， 而 这 些 函 数 所 
对 应 的 具体 函数 名 称 可 以 由 用 户 定义 


这 里 需要 注意 的 是 ， 表 13-3 中 所 定义 的 函数 名 称 是 可 以 更 改 的 ， 为 了 便于 理解 ， 建 议 读者 在 相 
应 的 名 称 上 加 上 后 缀 open、_close 及 _writer 等 。 

下 面 创建 一 个 基于 MySQL 数据 库 的 用 户 自 定义 处 理 器 ， 它 用 于 实现 将 用 户 的 会 话 信息 存储 在 
MySQL 数据 库 中 。 首 先 创建 一 个 数据 库 session， 然 后 在 该 数据 库 内 创建 表 sessioninfo。 有 具体 代码 如 
下 所 示 : 

案例 13-3 


mysql> CREATE DATABASE session; 
Query OK, 1 row affected (0.33 sec) 


mysql> USE session; 


其 次 ， 编 写 自 定义 会 话 处 理 器 的 相关 函数 及 其 代码 。sessionhandle.php 文件 用 于 存储 这 些 函数 
及 其 代码 ， 具 体 如 下 所 示 : 


岩 功 路 


哨 冯 3Mo0O 


到 
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testSessionHandle.php 页 面 为 了 测试 这 个 用 户 自 定义 会 话 处 理 器 的 实现 ， 它 的 具体 代码 如 下 
所 示 : 


把 上 述 文件 sessionhandle.php 和 testSessionHandle.php 存储 在 Apache 目录 的 子 目 录 htdocs\13 下 ， 
打开 下 浏览 器 ， 在 地 址 栏 中 输入 http://localhost/13/testSessionHandle.php， 按 Enter 键 ， 如 果 执 行 成 
功 将 看 到 如 下 所 示 的 信息 : 


这 时 ， 打 开 MySQL 客户 端 ， 然 后 查看 表 sessioninfo 中 的 数据 ， 具 体 如 图 13-5 所 示 。 

从 图 13-5 可 以 看 出 ， 上 述 案 例 向 表 
sessioninfo 中 插入 了 一 行 记录 ， 实 现 了 将 SD 
映射 到 会 话 变量 “JackLong”。 其 中 ， 这 条 记 
录 所 表示 的 信息 将 在 创建 之 后 的 1440s 到 期 ， 
因为 这 个 时 间 的 计算 是 通过 当前 时 间 加 上 


1440s 得 出 的 。 而 1440 是 在 php.ini 中 设 定 的 13-5 查看 表 sessioninfo 中 的 数据 
默认 到 期 时 间 ， 如 图 13-6 所 示 ， 用 户 可 以 根 
据 自己 的 需要 进行 修改 。 


小 


文件 D 入 加 呈 个 或 中， 喜 看 Yi 末了 00 


2 
Co 


; After hls nunber of secends，stored data vill be seen as “yarbaye” and 
; cleaned up by the garbage collection process. 
session.ge_naxlifetine = TAN 


i HOTE: IF you are using the subdirectory option For storing session files 

(see session.save_path above), then garbage collection does snot™ 
happen autonatically. you vill need to do your own garbage 
collection through a shell script, cron entry. or sone other netnod. 
For exanple, the Following script would is the equivalent of 
Setting session.ge_naxlifetine to 1440 (1449 seconds ~ 24 ninutes): 


图 13-6 默认 到 期 时 间 
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13.4 “会话 实例 


本 节 举 出 两 个 案例 ， 分 别 用 于 说 明 Cookie 和 会 话 处 理 Session 的 应 用 。 
13.4.1 ”Cookie 的 使 用 


本 案例 用 于 展示 Cookie 的 使 用 。 其 中 ， 用 户 在 访问 的 页 面 loginl.php 中 输入 自己 的 姓名 ， 然 后 
提交 到 LoginCookie.php 页 面 进行 处 理 。login1.php 页 面 的 具体 代码 如 下 所 示 : 


案例 13-4 
<html> 
<head> 
<meta http-equiv="Content-Type" content="text/html; charset=gb2312" /> 
<title> 设 置 用 户 Cookie 记录 的 姓名 数据 </title> 
</head> 
<body> 
<center> 
<table width="500" border="]1"> 
“Er 
<td> 
<form action="LoginCookie.php" method="post"> 
请 输入 你 的 姓名 : 
<input type="text" name="UserName"> 
<input type="submit"” value=" 确 定 提交 ”> 
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LoginCookie.php 页 面 用 于 将 访问 者 的 数据 存储 在 Cookie 中 ， 并 将 Cookie 中 的 数据 显示 在 网 页 
中 。 它 的 具体 代码 如 下 所 示 : 


</center> 
<body> 
</body> 
</html> 


把 上 述 文件 login.php 和 LoginCookie.php 存储 在 Apache 目录 下 的 htdocs\13 子 目录 下 ,打开 正 
浏览 器 ， 在 地 址 栏 中 输入 http://localhost/13/login1.php， 按 Enter 键 会 显示 如 图 13-7 所 示 的 页 面 。 
同 设 置 用 户 Cookie 记 


文件 并 牢 可 旭 ”下 看 中) 收 丫 生 工具 〇 帮助 加 
rr 


请 输入 你 的 姓名 。 宇 者 济 ] 


当 单 击 “ 硝 证 提交 ” 按 馈 时 ， 会 jserlame 的 值 ， 也 就 是 用 户 在 
文本 枉 中 箱 入 的 数据 ， 诠 交 结 | cokie.php 而 LogirCookie.phr 刚 
是 将 数据 写 入 用 户 计算 机 的 C 入 


ET 
图 13-7 loginl.php 页 面 


在 该 页 面 中 输入 用 户 姓 名 “ 宋 岩 岩 ”， 然 后 单 击 【 确 定 提交 】 按 钮 ， 会 显示 如 图 13-8 所 示 的 
页 面 。 


Sls Jam 


13.4.2 Session 的 使 用 Ee a 


目前 4UserNaveCookie 是 ， 宋 ; 
本 案例 用 于 展示 Session 的 使 用 。 其 中 ， 展 ee 
示 的 login2.php 页 面 用 于 收集 用 户 的 姓名 和 密 
码 ， 然 后 通过 单 击 【确认 提交 】 按 钮 提交 到 
LoginSession.php 页 面 进行 处 理 。 具 体 代码 如 下 图 13-8 ”LoginCookie.php 页 面 
所 示 : 


案例 13-5 
<html> 
<head> 
<meta http-equiv="Content-Type" content="text/html; charset=gb2312" /> 
<title> 输 入 用 户 的 姓名 和 密码 </title> 
</head> 
<body> 
<center> 
<table width="500" border="]1"> 
< 
<td> 
<form action="LoginSession.php" method="post"> 
<p> 请 输入 你 的 姓名 : 
<input type="text" name="username"></p> 
<p> 请 输入 你 的 密码 : 
<input type="password" name="pwd"></p> 
<input type="submit"” value=" 确 定 提交 "” > 


重新 给 入 用 产 性 
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LoginSession.php 页 面 用 于 判断 用 户 输入 的 姓名 和 密码 是 否 正确 , 如 果 都 正确 , 则 把 它们 存储 在 
Session 中 ， 并 显示 欢迎 消息 ;否则 显示 错误 消息 ， 请 求 重 新 登录 。 有 具体 代码 如 下 所 示 : 


把 上 述 文件 login php 和 LoginCookie php 存储 在 Apache 目录 下 的 htdocs\13 子 目录 下 ,打开 正 
浏览 器 ， 在 地 址 栏 中 输入 http://localhost/13/login2.php， 按 Enter 键 会 显示 如 图 13-9 所 示 的 页 面 。 


[TO 


We SeneTan 


醒 入 你 的 释 码 ， [ssesesse |] 
[要 要) 


图 13-9 login2.php 页 面 


在 该 页 面 中 输入 用 户 姓 名 “ 宋 岩 岩 ” 密码 “19820711”， 然 后 单 击 【确定 提交 】 按 钮 ， 会 显示 
如 图 13-10 所 示 的 LoginSession php 页 面 。 


alhost/13/Leginse， 


ETECTEEIRECITIETTIEII 


FTETTTTTTTTTT 
数据 已 经 顺利 存 俩 在 Seosi on 中 ， 用 户 成 功 登录 | 
SeneYan， 实 迎 你 访问 太 站 | 


图 13-10 成功 登录 的 LoginSession.php 页 面 
如 是 没有 使 用 上 述 的 用 户 和 密码 进行 登录 ， 会 显示 如 图 13-11 所 示 的 LoginSession.php 页 面 。 


p 一 看 icroeof Gx 


用 户 登录 失败! 


月 记 式 密码 销 涡 ， 清 和 新 如 录 1 
下 省 信用 记名 


图 13-11 登录 失败 的 LoginSession.php 页 面 
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第 14 章 用户 身份 验证 


学 习 目标 |oujecuve 


任何 成 功 的 Web 应 用 程序 安全 策略 的 基础 都 是 稳固 的 身份 验证 和 授权 手段 , 以 及 提供 机 密 数 据 
的 保密 性 和 完整 性 的 安全 通信 。 用 户 由 Web 应 用 程序 进行 身份 验证 ， 通 常 根据 用 户 名 和 密码 进行 ; 
随后 用 户 的 请 求 由 中 间 层 应 用 程序 服务 器 和 数据 库 服务 器 进行 处 理 ， 该 过 程 也 将 进行 身份 验证 以 便 
处 理 用 户 的 请 求 。 本 章 将 围绕 用 户 身份 验证 的 相关 内 容 进 行 介绍 ， 并 通过 案例 来 介绍 具体 是 如 何 实 
现 的 。 


区 内 容 摘要 |avstract 


理解 基本 的 HTTP 身份 验证 方式 

掌握 PHP 身份 验证 方式 
掌握 如 何 根据 需要 来 配置 数据 库 

掌握 如 何 向 数据 库 添 加 新 用 户 

掌握 用 户 登 录 的 实现 

理解 并 掌握 随 用 户 状态 而 改变 页 面 的 实现 
掌握 如 何 注销 用 户 

掌握 如 何 删除 用 户 


VvvvyvyvyvyYv 


14.1 Web 服务 器 提供 的 身份 验证 


身份 验证 是 Web 应 用 程序 最 常见 的 操作 。 通 常情 况 下 ， 两 种 身份 验证 方式 比较 常用 : 基本 的 
HTTP 身份 验证 和 PHP 身份 验证 。 下 面 将 分 别 进行 介绍 。 


14.1.1 基本 的 HTTP 身份 验证 


HTTP 协议 提供 了 一 种 非常 简单 、 有 效 的 用 户 验 证 方式 ， 也 就 是 
由 服务 器 来 咨询 资源 请 求 ， 其 中 客户 端 〈 浏 览 器 ) 用 于 提供 对 验证 过 
程 至 关 重 要 的 信息 。 该 验证 方式 的 具体 过 程 如 下 所 示 : 

(1) 客户 端 请 求 一 个 受 限 资源 。 

(2) 服务 器 用 一 个 401 (未 授权 访问 ) 响应 消息 对 这 个 请 求 做 出 
响应 。 

(3) 客 户 端 识别 401 响应 ,弹出 一 个 如 图 14-1 所 示 的 验证 对 话 框 。 
如 今 ， 许 多 浏览 器 都 提供 了 对 HTTP 验证 的 支持 ， 包 括 下 、Netscape 图 14-1 验证 对 话 框 


Navigator、Opera 和 Mozilla 等 。 

(4) 如 果 用 户 提供 了 相应 的 用 户 和 密码 ， 就 发 送 给 服务 器 进行 验证 ， 如 果 通 过 验证 ， 用 户 将 得 
到 请 求 访问 的 资源 ， 和 否则 将 拒绝 用 户 访问 。 

(5) 如 果 用 户 通过 了 验证 ， 浏 览 器 将 在 其 验证 缓存 中 存储 验证 信息 。 此 缓存 信息 将 一 直 保存 在 
浏览 器 中 ， 直 到 缓存 被 清空 ， 或 浏览 器 收 到 另 一 个 401 服务 响应 。 

HTTP 验证 可 以 有 效 地 控制 对 受 限 资源 的 访问 ， 但 它 不 能 保证 验证 信息 传输 通道 的 安全 。 也 就 
是 说 ， 对 于 位 置 合适 的 攻击 者 而 言 ， 偷 窍 或 监视 服务 器 与 客户 端 之 间 发 送 的 数据 是 很 容易 实现 的 。 
因为 用 户 提供 的 用 户 名 和 密码 都 包含 在 这 些 传输 数据 中 ， 并 且 没 有 经 过 任何 加 密 。 为 了 解决 这 种 方 
法 可 能 带 来 的 问题 ， 通 常情 况 下 采用 安全 套 接 字 (SSL) 来 实现 一 个 安全 的 通信 通道 。 目 前 ，SSL 
得 到 了 所 有 主流 Web 服务 器 的 支持 ， 包 括 本 书 用 到 的 Apache 和 微软 的 Internet 信息 服务 器 (IIS )。 


14.1.2 ”PHP 身份 验证 


PHP 身份 验证 方式 将 用 户 验证 直接 集成 到 Web 应 用 程序 逻辑 中 。 本 节 就 介绍 PHP 身份 验证 的 
相关 知识 及 实现 方法 。 
1. PHP 身份 验证 基础 知识 
PHP 使 用 两 个 预定 义 的 变量 来 验证 用 户 : $_SERVER[PHP AUTH USER] 和 $_SERVER 
[PHP AUTH PW']， 它 们 保存 了 用 户 验 证 所 需 的 用 户 名 和 密码 。 但 使 用 这 两 个 变量 时 应 注意 以 下 两 
个 方面 : 
。 两 个 变量 都 必须 在 每 个 受 限 页 面 的 开始 处 验证 ， 所 以 用 户 可 以 将 验证 代码 放 在 单独 的 文件 
中 ， 然 后 在 使 用 时 通过 REQUIREO 函 数 把 该 文件 包含 进去 即 可 。 
。 这 两 个 变量 在 CGI 版 本 的 PHP 中 不 能 正常 工作 ， 在 Microsoft IIS 上 也 不 起 作用 ， 具 体 说 明 
读者 可 查阅 相关 资源 。 
另外 ，PHP 在 处 理 验证 时 还 经 常用 到 两 个 标准 函数 : header0 和 issetO0。 下 面 对 这 两 个 函数 进行 
简单 介绍 。 
(1) headerO 
该 函数 用 于 向 浏览 器 发 送 原始 的 HITP 首部 。 它 的 使 用 格式 如 下 所 示 : 


int header ( string string [, bool replace [, int http response code]]) 


在 上 述 格式 中 ，string 参数 是 发 送 给 浏览 器 的 首部 信息 。 可 选 参数 replace 确定 此 信息 是 蔡 换 之 
前 发 送 的 首部 信息 , 还 是 和 以 前 的 首部 信息 一 起 发 送 。 可 选 参数 http_response_ code 用 于 定义 将 随同 
首部 信息 一 起 发 送 的 特定 响应 码 。 

(2) issetO 

该 函数 用 于 确定 是 否 已 为 一 个 变量 赋值 。 它 的 使 用 格式 如 下 所 示 : 


bool isset ( mixed Var [, mixed var [, .-..]]) 


在 上 述 格 式 中 ， 如 果 变 量 包含 值 则 返回 True， 和 否则 返回 False。 应 用 于 用 户 验证 时 ,该 函数 用 来 
确定 是 否 正 确 地 设置 了 $_ SERVER[PHP AUTH USER'] 和 $_SERVER['PHP_AUTH PW"] 变 量 。 
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2. 基于 验证 变量 的 身份 验证 

这 种 方法 通过 header0 函 数 发 送 HITP 首部 强制 进行 验证 ， 客 户 端 浏览 器 则 弹出 要 输入 用 户 名 
和 密码 的 对 话 框 ， 用 户 输入 相应 的 信息 并 提交 ， 这 时 用 户 输入 的 信息 将 被 传送 到 服务 端 之 后 保存 在 
PHP AUTH USER 和 PHP AUTH PW 这 两 个 全 局 变量 中 。 然 后 ， 利 用 这 两 个 变量 就 可 以 进行 用 户 
和 密码 的 验证 了 。 该 方法 的 实现 如 下 所 示 : 

案例 14-1 


<? 
if(!isset($ SERVER['PHP AUTH USER'])) 
上 
Header ('WWW-Authenticate: Basic realm=" 请 输入 用 户 和 密码 "'); 
Header ("HTTP/1.0 401 Unauthorized"); 
echo "本 站 提示 : 用 户 取消 了 登录 ! "; 
exit; 
1 
else 
| 
if ( !($ SERVER['PHP AUTH USER']=="bob" && $ SERVER['PHP AUTH PW']=="syy") 


{ 

// 如 果 是 错误 的 用 户 名 称 /密码 对 ， 强 制 再 验证 

Header(' WWW-Authenticate: Basic realm=" 请 重新 输入 用 户 和 密码 "' ) ; 
Header ("HTTP/1.0 401 Unauthorized") 7 

echo "本 站 提示 : 用 户 或 密码 错误 ! "; 

exit; 

} 


else 


{ 
echo "验证 成 功 ， 欢 迎 你 ! "; 


在 上 述 案例 中 ， 只 有 _SERVER['PHP_ AUTH _USER'] 和 _SERVER['PHP_AUTH_PW"] 分 别 被 设置 
为 bob 和 syy 后 才能 取得 所 请 求 的 资源 ， 否 则 将 强制 用 户 重新 输入 用 户 名 和 密码 ， 或 在 尝试 多 次 后 
输出 事先 设 定好 的 错误 消息 。 

3. 基于 文件 的 身份 验证 

基于 文件 的 身份 验证 就 是 利用 文本 文件 为 每 一 位 用 户 提供 唯一 的 登录 对 ， 这 样 就 可 以 记录 用 户 
特定 的 登录 时 间 、 活 动 和 动作 。 其 中 ， 该 文件 的 每 一 行 包含 一 个 用 户 名 和 加 密 密 码 对 ， 它 们 之 间 用 
冒号 (: ) 分 隔 。 

这 里 需要 注意 的 是 ， 该 文件 应 当 存 储 在 服务 器 文档 根 目录 之 外 ， 否 则 ， 攻 击 者 有 可 能 通过 一 些 
不 正当 手段 来 获取 该 文件 ， 从 而 造成 信息 的 泄露 。 当 然 用 户 可 以 使 用 不 对 称 密码 ， 以 明文 形式 存储 ， 
但 这 种 方式 极 不 安全 ， 因 为 如 果 文 件 权 限 配置 不 当 ， 那 么 能 够 访问 服务 器 的 用 户 就 可 以 查看 这 些 登 
录 信 息 。 

基于 文件 的 身份 验证 通过 将 文本 读 取 到 数组 中 ， 然 后 循环 处 理 数 组 ， 搜 索 匹配 的 数据 来 进行 身 
份 验证 。 在 这 个 过 程 中 需要 使 用 以 下 3 个 函数 : 


。 file(string filename) 该 函数 将 文件 读 取 到 数组 中 ,数组 的 每 个 元 素 分 别 包 括 文件 中 的 一 行 。 

。 explode(string separator,string string [.int limit]) 该 函数 将 字符 串 分 解 为 一 系列 字符 串 ， 每 
个 字符 串 的 边界 由 指定 的 分 隔 符 决定 。 

。 mds(sting str) 该 函数 使 用 RSA 数据 安全 公司 的 MD5 消息 摘要 算法 ( http://www.rsa.com ) 


来 计算 字符 串 的 MD5 散 列 值 。 
user.txt 文件 用 于 存放 前 面 介绍 的 登录 对 。 它 的 内 容 如 下 所 示 : 
案例 14-2 


bob:syy19820711 
jack:1jh19820624 
dear:love20070504 


txtlogin.php 文件 用 于 将 用 户 输入 的 信息 与 usertxt 文件 中 的 登录 对 进行 匹配 。 它 的 具体 代码 如 
下 所 示 : 


案例 14-2 
<?php 
$userfile = file("user.txt"); 


if (!(isset($ SERVER['PHP AUTH USER'])) && !(isset($ SERVER['PHP AUTH PW']))) 
{ 
Header ('WWW-Authenticate: Basic realm=" 请 输入 用 户 和 密码 "') ; 
Header ("HTTP/1.0 401 Unauthorized"); 
echo "本 站 提示 : 用 户 取消 了 登录 ! "; 
exit; 
} 
else 
{ 
foreach ($userfile as $login) 


list($username, $password) = explode(":", $login); 


$password = trim($password); 

if (($username == $ SERVER['PHP AUTH USER']) && ($password == $ SERVER 

['PHP_AUTH PW'])) 

. 
Sauthorized = TRUE; 
echo $_SERVER['PHP_AUTH USER']." 身 份 验证 成 功 ， 欢 迎 你 ! "; 
break; 

} 

else 
// 如 果 是 错误 的 用 户 名 称 /密码 对 ， 强 制 再 验证 
Header (' WWW-Authenticate: Basic realm=" 请 重新 输入 用 户 和 密码 "'); 
Header ("HTTP/1.0 401 Unauthorized"); 
echo "本 站 提示 :用 户 或 密码 错误 ! "; 


exit; 
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} 
} 
eS 
把 上 述 文件 txtlogin.php 存储 在 Apache 目录 的 htdocs\14 子 目录 下 ， 打 开 正 浏览 器 ， 在 地 址 栏 
中 输入 http://localhost/14/txtlogin.php， 按 Enter 键 会 显示 如 图 14-2 所 示 的 页 面 。 


图 14-2 基于 文件 身份 验证 的 登录 页 面 


在 该 页 面 中 输入 user.txt 文件 中 的 任 一 登录 对 这 里 使 用 第 一 行 登录 对 )， 单 击 【 确 定 】 按 钮 ， 
成 功 通过 验证 后 将 显示 如 下 所 示 的 信息 : 


bob 身份 验证 成 功 ， 欢 迎 你 ! 

如 果 此 时 单 击 【 取 消 】 按 钮 将 显示 如 下 所 示 的 信息 : 

本 站 提示 : 用 户 取消 了 登录 ! 

如 果 重 复 输入 几 次 错误 的 用 户 名 或 密码 ， 将 显示 如 下 所 示 的 信息 : 
本 站 提示 :用 户 或 密码 错误 ! 


这 里 文本 文件 usertxt 中 的 密码 为 明文 的 形式 ， 如 果 密 码 是 以 MD5 加 密 后 的 形式 存在 的 ， 那 么 
将 上 述 代码 中 的 语句 〈$password 一 $_ SERVER[PHP AUTH PW]) 修改 为 如 下 所 示 即 可 : 


Spassword == md5($ SERVER['PHP AUTH PW']) 


4. 基于 数据 库 的 身份 验证 

当 身 份 验证 要 处 理 很 多 用 户 ， 或 者 经 常 对 用 户 进行 增加 、 删 除 、 修 改 时 ， 以 上 两 种 身份 验证 方 
法 就 不 适合 了 ， 这 时 基于 数据 库 的 身份 验证 将 是 一 个 很 好 的 解决 方案 ， 因 为 它 不 仅 解决 了 管理 的 方 
便 性 ， 提 高 了 可 扩展 性 ， 而 且 可 以 集成 到 一 个 更 大 的 数据 库 中 。 

基于 用 户 的 身份 验证 通过 对 用 户 表 执行 一 个 查询 ， 使 用 所 输入 的 用 户 名 和 密码 作为 查询 条 件 ， 
如 果 从 数据 库 中 查找 到 相关 的 记录 ， 那 么 该 用 户 通过 身份 验证 ， 否 则 给 出 相关 的 报错 消息 。 

5. 基于 IP 的 身份 验证 

在 基于 数据 库 的 身份 验证 中 ， 由 于 用 户 名 和 密码 有 可 能 会 被 黑客 或 一 些 别有用心 的 人 获取 ， 所 
以 为 了 解决 这 个 问题 ， 进 一 步 保 证 用 户 及 其 信息 的 安全 ， 一 种 有 效 的 方法 不 仅 需 要 合法 的 用 户 名 和 
密码 登录 对 ， 还 需要 一 个 与 此 相关 联 的 下 地址。 这 样 即使 用 户 名 和 密码 被 泄露 ， 得 到 它 的 人 也 无 法 
通过 身份 验证 。 


基于 了 P 的 身份 验证 的 实现 过 程 与 基于 数据 库 的 身份 验证 的 实现 过 程 基本 相同 , 它 除了 使 用 所 输 
入 的 用 户 名 和 密码 作为 查询 条 件 外 , 还 需要 匹配 事先 设 定 的 与 之 相关 联 的 人 地址。 读者 可 以 根据 基 
于 数据 库 的 身份 验证 过 程 加 以 改进 来 实现 该 方法 。 


14.2 ”实现 用 户 的 身份 验证 中 


本 节 将 介绍 如 何 通 过 配置 用 户 数据 库 来 处 理 登录 、 向 数据 库 中 添加 新 的 用 户 、 数 据 库 中 己 有 用 
户 进 行 登录 、 更 新 用 户 登录 后 的 页 面 、 注 销 已 经 登录 的 用 户 以 及 删除 不 再 需要 的 用 户 。 


14.2.1 ”配置 数据 库 来 处 理 登 录 


这 里 使 用 MySQL 5.0 创建 数据 库 ， 名 称 为 users， 然 后 在 users 数据 库 中 创建 表 myuser， 该 表 主 
要 用 来 保存 用 户 的 登录 名 称 、 真 实 姓名 、 密 码 、 电 子 邮 件 、 联 系 电话 和 通信 地 址 。 该 表 中 的 用 户 在 
登录 成 功 后 可 以 执行 相应 的 操作 。 它 的 具体 说 明 如 表 14-1 所 示 。 
表 14-1 用 户 表 myuser 
字段 名 备注 
| 否 ”| 用 户 ID， 自动 增加 
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ame 一 ea as 用 户 名 称 
mame | vacha(y) | 是 | 不 实 姓名 
Ipwd | vaehadmy | 吾 | 用 户 密码 
lemail | vachaG%) | 是 | 电子 邮件 
Iphone | vacha(y) | 是 | 联系 电话 


| van | 是 | 通信 地 址 


下 面 使 用 phpMy Admin 创建 数据 库 及 表 ， 具 体 过 程 如 下 所 示 : 

(1) 由 于 前 面 已 经 对 phpMyAdmin 进行 了 配置 ， 所 以 这 里 直接 打开 正 浏览 器 ， 在 地 址 栏 中 输 
入 http://localhost/phpmyadmin/index.php， 如 果 执 行 成 功 会 显示 phpMyAdmin 初始 页 面 ， 否 则 需要 根 
据 前 面 的 介绍 检查 相应 的 配置 ， 直 到 成 功 执 行 phppMyAdmin 初始 页 面 ， 如 图 14-3 所 示 。 


laddress 


i Er 
phpMyAdmin wiki 

俐 phpMyAdmin 官方 网 站 

» [ChangeLog] [Subversion] [Lists] 


utf8_general_ci 


匠 创 建 一 个 新 的 数据 库 加 


Users 


gb2312_chinese_ci 司 [ 创建 
于 显示 MySQL 的 运行 信息 


本 地 Intranet 


图 14-3 index.php 页 面 
(2) 在 【创建 一 个 新 的 数据 库 】 下 面 输入 数据 库 名 “users”， 并 在 下 面 的 下 拉 列 表 框 中 选择 
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gb2312_chinese ci 选项 ， 单 击 【创建 】 按 钮 ， 创 建 数据 库 users， 执 行 成 功 后 如 图 14-4 所 示 。 


当 Localhost / localhozt / uaere | phplyhdain 2. 10-1 一 icrosoft Internet Erploror 


EECTIEEOREEII 
地 址 0 | 阐 http:/1localhostvyphpmradninyintec pag 
田 服务 器 : localhost ， 县 数据 库 : users 


盏 结构 ”如 SOL 己 按 看 加 查询 总 导出 县 Import 稀 操 作 ， 多 权限 
加 出 除 


数据 库 中 没有 夫 。 


独 在 数据 库 users 中 创建 一 个 新 表 一 一 一 一 一 一 一 一 一 一 一 一 一 一 一 
名 宇 :|myuser | Number offields: 17| | 


图 14-4 在 users 数据 库 中 创建 表 


(3) 在 “名 字 ” 文 本 框 中 输入 “myuser”; 在 Number of fields 文本 框 中 输入 “7”， 即 表 中 有 7 
个 字段 。 然 后 单 击 【 执 行 】 按 钮 创建 表 myuser 的 结构 ， 如 图 14-5 所 示 。 


/ Pe phplyle 
文件 0】 奖 辑 E) 全 看 中 收 冶 以 工具 CD 裙 凤 0 : 
下 he:/1localhosvVphpnyaaninyinter phE 
中 服务 器 : localhost ， 永 数据 库 : usars ， 目 表 : myuser 
一 同 国 司 可 一 类 型 整理 
1d [NT 312_chinese_cl 


Imame VARCHAR “ 司 ] _chinese_ci 


mame [vaRCHAR 1 gb2312_chinese_cl 


lpwd [vARCHAI 1 Ma chinese_ci 
lomail VARCHAR | [gb2312_chinese_ci 
Iphone |vARCHAR 2 job2312_chinese_ci 
laddress VARCHAR “ 司 gb2312_chinese_ci 


表 注 释 : Storage Engine: 整理 
加 


InoDB 闭 。 9b2312_chinese_cl 


保有 | 或 沫 [1 | 字段 [玉生 


图 14-5 创建 表 myuser 的 结构 


(4) 在 该 页 面 中 , 按 前 面 设 定 的 表 结 构 进行 输入 ,并 设置 id 字段 的 其 他 属性 为 UNSIGNED (无 
符号 )、auto_increment (自动 增加 ) 和 Primary Key (主键 )。 具 体 如 图 14-6 所 示 。 


| auto_increment 六 


图 14-6 设置 i 字段 的 其 他 属性 


(5) 设置 完成 后 ， 单 击 【保存 】 按 钮 创建 表 myuser。 如 果 创 建 的 表 不 符合 用 户 的 要 求 ， 可 以 对 
其 进行 再 次 修改 ， 直 到 满意 为 止 。 至 此 ， 用 户 身 份 验证 用 到 的 数据 库 及 表 都 已 创建 完成 ， 下 面 就 可 
以 进行 相应 的 操作 了 。 


14.2.2 ”添加 新 的 用 户 


添加 新 用 户 时 ， 当 用 户 输入 相应 的 信息 后 ， 单 击 【提交 】 按钮 ， 确 认 进行 添加 。 这 时 ， 如 果 要 
求 必 须 输入 的 信息 没有 输入 或 两 次 输入 的 密码 不 一 致 ， 系 统 都 将 给 出 提示 消息 。 添 加 成 功 后 系统 将 
返回 相应 的 消息 ， 如 果 当 前 提交 的 信息 中 用 到 的 用 户 名 称 已 经 存在 ， 那 么 添加 失败 将 返回 “当前 用 
户 添加 失败 ! 已 经 存在 同名 的 用 户 ， 请 另 选 其 他 用 户 名 称 重新 注册 ”的 消息 。 

adduserphp 页 面 用 于 收集 用 户 的 信息 。 它 的 具体 代码 如 下 所 示 : 


串 汕 妆 性 悄 沿 19SAW+dHd 


adduserok.php 页 面 用 于 将 由 adduserphp 页 面 传递 过 来 的 用 户 信息 插入 表 myuser 中 。 注 意 ， 这 
里 在 插入 记录 之 前 进行 了 检查 ， 如 果 存 在 相同 的 用 户 名 称 ， 则 插入 失败 。 它 的 具体 代码 如 下 所 示 : 


$lemail = $ POST['lemail']; 
$lphone = $ POST['lphone']; 
$laddress = $ POST['laddress']; 


$db = mysql connect ("localhost",root,123456789); 
if(!$db) 
{ 

echo "出 错 了 : 不 能 连接 MysQL server!"; 

} 


mysql select db("users"); 
$query = "SELECT * FROM myuser WHERE lname = '$lname'"; 


$result = mysql query ($query); 
$rows = mysql num rows ($result); 
if($rows>0) 
{ 

echo "添加 新 用 户 失败 ! <hr><p>"; 

echo "用 户 ; [".$rname."]"; 

echo "已 经 有 相同 的 用 户 名 称 存在 ! "; 

echo "<br> 请 返回 ，<a href='adduser .php'> 重 新 输入 用 户 名 称 </a>"; 

Sse 
{ 
$query = "INSERT INTO myuser (lname, rname,1lpwd, lemail,lphone,1laddress)" 
."VALUES ('$lname','$rname','$lpwd','$lemail','$lphone','$laddress')"; 
$result = mysql query ($query); 
$id = mysql insert id(); 
if ($id>0) 
{ 
echo "添加 新 用 户 成 功 ! <hr><p>"; 
echo "新 添加 的 用 户 是 : ".$rname."<p>"; 
} 
else 


{ 


丙 邮 全 如 二 入 ” 需 六 潍 


echo "添加 新 用 户 失败 ! <hr><p>"; 
echo "用 户 : [".$rname."]"; 
echo "添加 失败 ! "; 
} 
echo "<br><a href='adduser .php'> 重 新 添加 用 户 </a>"; 


} 
?> 
</center> 
</body> 
</html> 


把 上 述 文件 adduserphp 和 adduserok.php 存储 在 Apache 目录 下 的 htdocs\14 子 目录 下 ,打开 正 
浏览 器 ， 在 地 址 栏 中 输入 http://localhost/14/adduser.php， 按 Enter 键 会 显示 如 图 14-7 所 示 的 页 面 。 
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当 添加 下 的 用 户 -icrosoft Internet Fzplorer 
Er 
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箱 入 你 移 信 息 ， 带 + 号 项 双 须 填写 ， 


14-7 添加 用 户 adduserphp 页 面 


在 该 页 面 中 输入 用 户 名 称 “syy” 真实 姓名 “ 宋 岩 岩 ” 密码 “syy” 等 用 户 信息 ， 然 后 单 击 【 提 
交 】 按 钮 ， 如 果 添 加 成 功 会 显示 如 图 14-8 所 示 的 页 面 。 
涛 加 新 用 户 的 执行 结果 Wicrosoft Internet Explorer 


文件 F) 风 句 F) 查看 W) 收藏 和 ) 工具 T) 帮助 00 
地 址 四 ) | 着 http://1ocwlhost/14/ ddaserok php v 回 Wl i ” 


添加 新 用 户 成 功 | 


新 添加 的 用 户 是 ， 宋 岩 岩 


重新 添加 用 户 


ETT 


图 14-8 成 功 添加 新 用 户 


14.2.3 ”登录 用 户 


在 本 案例 中 ，login.php 页 面 用 于 让 用 户 输入 用 户 名 称 和 密码 ， 并 提交 给 loginchk.php 页 面 进行 
处 理 。 它 的 具体 代码 如 下 所 示 : 


案例 14-4 
<html> 
<head> 
<meta http-equiv="Content-Type" content="text/html; charset=gb2312" /> 
<title> 用 户 登 录 </title> 
</head> 
<body> 
<center> 
<table width="500" border="1"> 
> 
<td> 
<form action="loginchk.php" method="post"> 
<div align="center"><strong> 用 户 登 录 
</strong></div> 
<hr> 


<p align="center"> 请 输入 你 的 用 户 名 : 


loginchk.php 页 面 用 于 验证 用 户 的 用 户 名 称 和 密码 是 否 正 确 。 如 果 用 户 名 称 和 密码 与 myuser 表 
中 的 某 一 条 记录 相对 应 ， 那 么 就 将 该 用 户 的 名 称 和 真实 姓名 保存 到 Session， 以 备 后 面 操 作 的 需要 ， 
否则 返回 相应 的 报错 消息 。 该 页 面 的 具体 代码 如 下 所 示 : 


$_SESSION['orname'] = $rname;echo "用 户 登录 成 功 ! <hr><p>"; 
echo "登录 用 户 是 : ".$rname."<p>"; 
echo "<br> 现 在 你 可 以 <a href='usercenter .php'> 转 到 用 户 中心 </a>"; 
} 
else 
‘ 
echo "用 户 登录 失败 ! <hr><p>"; 
echo "用 户 密码 不 正确 ! "; 
echo "<br><a href='login.php'> 请 重新 登录 </a>"; 


else 
也 
十 echo "用 户 登录 失败 ! <hr><p>"; 
去 echo "当前 用 户 不 存在 ! "; 
了 echo "<br><a href='adduser .php'> 请 先 注 册 ， 再 进行 登录 ! </a>"; 
名 上 
到 ?> 
2 </center> 
国 </body> 
习 </html> 
手 
册 把 上 述 文件 login.php 和 loginchk.php 存储 在 Apache 目录 下 的 htdocs\14 子 目录 下 ,打开 正 浏 览 


器 ， 在 地 址 栏 中 输入 http://localhost/14/login.php， 按 Enter 键 会 显示 如 图 14-9 所 示 的 页 面 。 


文公 0) 纲 久 IE) 可 看 W) 收 意 Ih) 工具 m) 末 且 0 
堆 站 四 | 图 htt scahosttwlegin piy 


用 户 登 录 


请 往 入 你 的 用 户 各 [a7 
请 箱 入 你 的 密码 。 。"。 
| 本 


图 14-9 用 户 登 录 窗 口 


在 该 页 面 中 输入 用 户 名 称 “syy”、 密 码 “syy” 等 用 户 信息 ， 然 后 单 击 【提交 ]】 按 钮 ， 如 果 登 录 
成 功 会 显示 如 图 14-10 所 示 的 页 面 。 


全 关 加 新 用 户 的 执行 结果 - Wicrosoft Internet Explorer 
文 省) 名 名 全 ) 查看 W 收 京 ) 工具 0) 帮助 00 
庙 泪 0) 恒 http//1ocslhost/14/1oginchk php 


用 户 登录 成 功 ! 


登录 用 户 是 : 宋 岩 岩 
现在 你 可 以 转 到 用 户 中 心 


图 14-10 成功 登录 后 的 页 面 
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14.2.4 ”更 新 需要 用 户 登 录 的 页 面 


在 网 络 上 ， 很 多 网 站 为 注册 用 户 提供 了 许多 个 性 化 功能 ， 当 用 户 访问 该 类 型 的 网 站 而 并 没有 注 
册 为 该 类 型 网 站 的 用 户 时 ， 这 些 功 能 对 该 用 户 来 说 是 透明 且 不 可 用 的 。 例 如 ， 常 见 的 网 上 购物 就 是 
这 样 的 ， 只 有 注册 为 该 购物 网 站 的 用 户 ， 并 在 选 购 商品 前 成 功 登录 才能 进行 相应 的 操作 。 

下 面 的 案例 用 来 模拟 该 功能 ， 当 用 户 登 录 成 功 后 可 以 修改 自己 的 资料 、 密码、 发 表 日 志和 图 片 ， 
以 及 对 它们 进行 管理 等 。 否 则 将 只 能 查看 一 些 普通 的 信息 。usercenterphp 文件 用 于 实现 该 功能 ， 其 
有 具体 代码 如 下 所 示 : 


PHP+MySQL 了 不 种 钊 民 慎 训 


“E> 
<td> 
<p> 在 网 络 上 , 很 多 网 站 为 注册 用 户 提供 了 许多 个 性 化 功能 ， 当 用 户 访问 该 类 型 的 网 站 而 并 没有 注册 为 
该 类 型 网 站 的 用 户 时 ， 这 些 功能 对 该 用 户 来 说 是 透明 且 不 可 用 的 。 例 如 ， 常 见 的 网 上 购物 就 是 这 样 的 ， 
只 有 注册 为 该 购物 网 站 的 用 户 ， 并 在 选 购 商品 前 成 功 登 录 才 能 进行 相应 的 操作 。 
下 面 的 案例 用 来 模拟 该 功能 ， 当 用 户 登 录 成 功 后 可 以 修改 自己 的 资料 、 密 码 、 发 表 日 志和 图 片 ， 以 及 对 
它们 进行 管理 等 。 否 则 将 只 能 查看 一 些 普通 的 信息 。 
(<span class="STYLE1"> 此 处 的 信息 用 户 登录 与 否 都 能 看 到 ) </span></p> 
</td> 
</tr> 
</table> 
</center> 
</body> 
</html> 


把 上 述 文件 usercenter.php 存储 在 Apache 目录 下 的 htdocs\14 子 目 录 下 ， 打 开 正 浏览 器 ， 在 地 
址 栏 中 输入 http://localhost/14/usercenter.php， 按 Enter 键 会 显示 如 图 14-11 所 示 的 页 面 。 


ERROREZI EE Cz 
地 二 阐 Mws /fechaeeente ply 


用 户 登录 


请 输入 你 的 用 户 名 ， 请 输入 你 的 字 码 ， 


在 网 络 上 ， 很 多 网 站 为 注册 用 户 提供 许多 个 性 化 的 功能 ， 
网 王后 AS 人 这 些 功能 对 该 用 户 未 党 ; 识 半 站 和 例如 , 党 
A ， 并 在 选 购 商品 前 记 功 登录 才能 
站 a ue 拒 们 就 来 “ 当 用 户 登录 成 功 后 可 以 修改 自 
的 次 村、 发 和 否则 将 只 能 查看 一 些 普通 的 信 
前 Wd 


图 14-11 用 户 未 登录 时 的 页 面 
在 该 页 面 中 输入 用 户 名 称 “syy”、 密 码 “syy” 等 用 户 信息 ， 然 后 单 击 【提交 】 按 钮 ， 如 果 登 录 
成 功 会 显示 如 图 14-10 所 示 的 页 面 。 然 后 单 击 【 转 到 用 户 中心 】 超 级 链接 会 显示 如 图 14-12 所 示 的 
页 面 。 


ETcamgcrgecogca 
Ms 


用 户 管理 中 心 
Fi 岂 用 户 名 称 ，[ syy] 真实 
您 改 资料 修改 季 码 
注册 十 | re 


尖 用 户 

明 的 、 而 不 可 使用 的 。 训 bn， 常 

We 
户 查 录 成 蕊 ) Ra 可 


图 14-12 用 户 成 功 登 录 后 的 页 面 
14.2.5 ”注销 用 户 


通常 情况 下 ， 当 用 户 登 录 成 功 以 后 ， 为 了 保护 用 户 个 人 信息 的 安全 ， 网 站 管理 者 建议 用 户 在 离 
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开 网 站 时 注销 登录 ， 而 不 是 等 Session 自动 失效 ， 因 为 这 样 可 能 会 给 一 些 别 有 用 心 的 人 以 可 乘 之 机 。 
logout.php 文件 用 于 实现 该 功能 ， 其 具体 代码 如 下 所 示 : 


用 户 名 称 : [ 
<?php 
echo S$olname; 
?2>] 
真实 姓名 : [ 总 
<?php 局 
echo $orname; ~ 
$_SESSION['olname'] 
$ SESSION['orname '] = ""; 
?>] 
<br><b> 操 作 结果 : </b> 注 销 用 户 成 功 ! 

<br> 我 要 从 : 

<a href="]ogin.php"> 登 录 页 面 </a> 

<a href="usercenter.php"> 用 户 管理 中 心 </a> 登 录 

<?php 

中 

?> 
</div> 
</body> 
</html> 


把 上 述 文件 usercenterphp 存储 在 Apache 目录 下 的 htdocs\14 子 目录 下 ， 打 开 正 浏览 器 ， 在 地 
址 栏 中 输入 http://localhost/14/logout.php， 按 Enter 键 会 显示 如 图 14-13 所 示 的 页 面 。 
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文 仲 ) 蚁 可 于 ) 查看 MW) 收 本 0) 工具 CF) 帮助 00 
0 科 hp/1oedlhost /ifloeont ah 


你 还 没有 进行 登录 ， 所 以 不 能 执行 此 操作 ， 请 先 登 录 ! 
我 要 登录 


PETE 


14-13 用户 未 登录 时 查看 logoutphp 页 面 


单 击 【 我 要 登录 】 超 级 链接 会 显示 login.php 登录 页 面 ， 然 后 输入 用 户 名 称 “syy” 密码 “syy”， 
成 功 登 录 后 用 户 可 以 通过 单 击 【 转 到 用 户 中 心 】 超 级 链接 打开 用 户 管理 中 心 usercenterphp 页 面 ， 这 
时 单 击 【 退 出 登录 】 超 级 链接 ， 会 显示 如 图 14-14 所 示 的 退出 登录 页 面 。 


ET 收 蒜 的。 工具 C) 帮助 00 
起 址 站) 个 http:Javealhost/la/logoat php 


。 福 销 用 户 。 放 户 告 称 ，[ syy] 真 去 姓 宇 ，[ 末 涯 省] 
操作 结果 :注销 月 户 所 功 | 


图 14-14 退出 登录 页 面 
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14.2.6 ”删除 用 户 


当 某 些 用 户 不 再 需要 时 ， 应 该 将 其 删除 。 例 如 ， 公 司 的 OA 办 公 系 统 ， 当 某 员工 离开 公司 后 ， 
那么 他 的 用 户 将 被 删除 ， 因 为 他 的 离开 致使 他 失去 了 登录 该 系统 的 权利 。 下 面 的 案例 将 介绍 如 何 删 
除 某 一 用 户 。 

deluserphp 页 面 用 于 显示 所 有 的 用 户 及 其 信息 ， 当 前 登录 的 用 户 可 以 删除 其 他 用 户 账号 ， 但 不 
能 删除 自己 的 账号 。 通过 单 击 每 条 记录 后 面 的 【删除 】 超 级 链接 提交 给 deluserok .php 页 面 进行 处 理 。 
该 页 面 的 具体 代码 如 下 所 示 : 


串 疏 名 性 由 浊 19SAW+dHd 


deluserok.php 页 面 用 于 执行 删除 操作 ， 它 根据 deluserphp 页 面 传递 过 来 的 用 户 编号 ID 查询 数 
据 ， 如 果 找 到 则 将 其 删除 ， 否 则 显示 报错 消息 。 该 页 面 的 具体 代码 如 下 所 示 : 


if (Snum>0) 
下 

echo "删除 用 户 成 功 ! <hr><p>"; 
echo "删除 用 户 的 编号 是 : " .$id."<p>"; 
else 

和 

echo "删除 用 户 失败 ! <hr><p>"; 

echo "用 户 编号 : [".$rname."]"; 
echo "添加 失败 ! "; 

echo "<br><a href='deluser .php'> 重 新 删除 用 户 </a>"; 

} 


2 
</center> 
</body> 
</html> 


把 上 述 文件 deluserphp 和 deluserok.php 存储 在 Apache 目录 下 的 htdocs\14 子 目录 下 ,打开 正 
浏览 器 ， 在 地 址 栏 中 输入 http:Wlocalhost/14/deluserphp， 按 Enter 键 会 显示 如 图 14-15 所 示 的 页 面 。 
翌 选择 要 删除 用 户 - icrosoft Intprnet Explorer 


文件 0) 久久 吕 ， 全 看 MW] 愧 窟 扩 工 具 (T) 六 册 00 
此 二 四) | 痢 http /1loealhosVt4ydtlaxer phg 


当前 用 户 :了 


本 站 提示 

你 还 役 有 登录 ， 元 权 查 看 该 页 面 

请 先 登录 

用 户 编号 用 户 名 称 真实 姓名 用 户 密码 电子 邮件 联系 电话 家 庭 地 址 操作 


ET 


图 14-15 未 登录 用 户 查看 deluserphp 页 面 


单 击 【 请 先 登录 】 超 级 链接 会 显示 login.php 登录 页 面 ， 然 后 输入 用 户 名 称 “syy”、 密 码 “syy”， 
成 功 登 录 后 用 户 可 以 通过 单 击 【 转 到 用 户 中 心 】 超 级 链接 打开 用 户 管理 中 心 usercenterphp 页 面 ， 这 
时 单 击 【 管 理 用 户 】 超 级 链接 ， 会 显示 如 图 14-16 所 示 的 用 户 管 理 页 面 。 


当前 用 户 ，[ 宋 岩 岩 ] 


用 户 编号 用 户 名 称 直 实 姓名 用 户 窗 码 。 电子 邮件 。 ”联系 电话 | 家 庭 地 址 操作 


国 纪 | 


lzx9codi. cn ”01023530844 | 北京 市 丰台 区 击 除 
EE 


14-16 用 户 管理 deluserphp 页 面 


第 
14 
章 
用 
户 
身 
份 
验 
证 


卫 
ue 
sl 
十 

< 

op) 
oO 
re 
完 
全 
学 
习 
手 
册 


在 该 页 面 中 ， 如 果 要 删除 用 户 编号 为 6 的 记录 ， 可 以 单 击 该 记录 后 面 的 【删除 】 超 级 链接 ， 会 
显示 如 图 14-17 所 示 的 页 面 ， 这 时 再 返回 用 户 管理 页 面 此 条 记录 将 不 再 显示 ， 因 为 它 已 经 被 删除 。 


马 一 除 用 户 结果 - 了 crosoft Irternet Ezplorer 局 回民 
Er E73 
四 | 局 wmacalesvtwiaaserec pyrit3 已 


到 除 用 户 上 成功! 


删除 用 户 的 编号 是 ，6 


图 14-17 执行 删除 操作 页 面 


说 明 : 以 上 各 小 节 的 案例 组 成 了 一 个 完整 而 简单 的 用 户 管理 系统 ， 在 对 它们 进行 操作 时 ， 应 注 
意 它 们 之 间 是 相互 关联 的 。 例 如 ， 当 用 户 未 登录 时 ， 某 些 页 面 是 不 允许 用 户 查 看 的 ， 如 果 查 看 ， 它 
将 给 出 报错 信息 ， 并 引导 用 户 先进 行 登录 或 注册 。 另 外 ， 上 述 案例 涉及 对 数据 库 的 常用 操作 ， 读 者 
应 熟练 掌握 。 


第 15 章 PHP 和 XML 


= 学 习 目标 |oujecuve 


XML 是 W3C 的 标准 ， 主 要 用 来 存储 数据 和 交换 数据 。 大 部 分 语言 都 支持 XML， 如 Java、C++ 
等 。 PHP 对 XML 的 支持 更 加 强大 ， 不 仅 实现 了 XML 的 所 有 流行 接口 ， 如 DOM、SAX 等 ， 还 内 置 
SimpleXML 函数 库 ， 方 便 初学 者 解析 XML 文档 。 本 章 将 从 XML 的 语法 开始 ， 详 细 介 绍 PHP 对 
XML 文档 的 解析 ， 并 简单 介绍 在 客户 端 解析 XML 文档 。 


SS 
2 内 容 摘要 |avstrsct 


了 解 XML 的 产生 原因 和 优点 
熟练 掌握 编写 规范 的 XML 文档 

掌握 PHP 命名 空间 的 使 用 

掌握 编写 有 效 的 XML 文档 

了 解 XML 文档 的 相关 技术 

了 解 PHP 解析 XML 文档 的 方法 

熟练 掌握 在 PHP 中 使 用 DOM 处 理 XML 
熟练 掌握 在 PHP 中 使 用 SimpleXML 处 理 XML 
掌握 在 客户 端 处 理 XML 


Vvvvyvyvyv 


vv 


15.1 XML 


目前 ， 在 Web 程序 的 开发 方面 ， 如 传输 数据 、 存 储 数据 、 配 置 服务 器 都 可 以 使 用 XML。XML 
还 可 以 做 成 一 个 单独 的 Web 页 面 ， 显 示 在 客户 端 ， 从 而 达到 数据 和 显示 的 分 离 。 同 样 ，PHP 作为 一 
门 主流 的 Web 技术 ， 对 XML 的 支持 是 非常 彻底 的 。 本 节 将 从 XML 的 基础 知识 讲 起 ， 阅 述 PHP 技 
术 中 XML 的 使 用 。 


15.1.1 XML 概述 


XML (eXtensible Markup Language， 可 扩展 标记 语言 ) 是 标记 语言 的 一 种 ， 是 W3C 标记 语言 
的 标准 。XML 的 前 身 是 SGML (The Standard Generalized Markup Language)， 是 自 IBM 从 20 世纪 
60 年 代 开 始 发 展 的 GML (Generalized Markup Language) 标准 化 后 的 名 称 ，1978 年 ，ANSI 将 GML 
加 以 整理 规范 , 发 布 成 为 SGML。 XML 的 产生 是 为 了 弥补 SGML 语言 的 不 足 。 该 语言 具有 和 HTML 
语言 相似 的 语法 ， 并 且 比 HTML 语言 有 更 好 的 可 扩展 性 。 所 谓 可 扩展 性 是 指 XML 人 允许 用 户 按照 
XML 规则 自 定义 标记 。 这 些 标记 可 以 自由 定义 ,其 目的 是 使 XML 文件 能 够 更 好 地 体现 数据 的 结果 
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和 含义 。 

XML 是 一 种 描述 数据 和 数据 结构 的 元 标记 语言 , 可 以 保存 在 任何 可 以 存储 文本 的 文档 中 。 该 语 
言 是 元 (一 族 ) 语言 而 不 是 一 种 语言 ， 元 语言 是 一 种 本 身 能 够 创建 一 种 语言 的 语言 ， 是 一 种 可 以 上 
来 创建 自己 标记 的 标记 语言 。 它 是 用 来 描述 其 他 语言 的 语言 ， 它 允许 自己 设计 标记 ,如 HTML 标记 
语言 ， 使 用 HTML 标记 创建 一 个 HTML 页 面 ， 该 页 面 在 多 个 标记 的 有 机 组 织 下 ， 只 能 以 一 种 格式 
显示 ， 如 有 段落 标记 <p>， 无 论 在 任何 一 个 页 面 ， 其 代表 的 含义 是 不 变 的 。 但 是 在 XML 标记 语言 中 ， 
可 以 用 中 文 < 段落 > 来 代表 段落 的 含义 ， 也 可 以 使 用 <pp> 代 表 段 落 的 含义 ，XML 是 一 种 能 够 设计 出 
自己 需要 的 多 个 标记 的 标记 语言 。 

XML 是 一 种 基于 文本 的 格式 ,可 以 使 用 任何 一 个 文本 编辑 器 编辑 XML 文档 。 在 许多 方面 类 似 
于 HTML， 后 者 是 专 为 存储 和 传输 数据 而 设计 的 。XML 源 是 由 XML 标记 组 成 的 ， 每 个 XML 标记 
包括 一 个 开始 标记 (<title>), 一 个 结束 标记 (</title>) 以 及 两 个 标记 之 间 的 信息 ( 称 为 内 容 )。 就 像 HTML 
一 样 ，XML 文档 保存 利用 标记 注释 的 文本 。 然 而 ， 与 HTML 不 同 的 是 ，XML 人 允许 无 限 的 标记 集 ， 
各 个 标记 集 并 不 表示 如 何 显示 ， 而 是 表示 其 含义 。 例 如 ， 可 以 将 XML 标记 定义 为 价格 、 订 单 编号 
或 名 称 。 由 文档 的 作者 确定 使 用 何 种 数据 以 及 哪 种 标记 名 称 最 合适 。 

XML 文件 重点 解决 的 是 XML 的 数据 存储 和 传输 ， 而 不 是 这 些 数据 的 显示 样式 。 可 以 利用 一 个 
程序 来 获取 XML 文档 中 的 数据 ， 并 利用 这 些 被 获取 的 数据 进行 操作 。 需 要 注意 的 是 ，XML 是 用 来 
描述 数据 的 ， 重 点 是 : 什么 是 数据 ， 如 何 存 放 数 据 。 

XML 是 HTML 的 补充 , 但 XML 并 不 是 HTML 的 蔡 代 品 。 在 将 来 的 网 页 开发 中 ，XML 将 被 用 
来 描述 、 存 储 数据 ， 而 HTML 则 是 用 来 格式 化 和 显示 数据 的 。 对 于 XML 最 好 的 形容 可 能 是 : XML 
是 一 种 跨 平台 的 ， 与 软 、 硬 件 无 关 的 ， 处 理 信息 的 工具 。 

XML 自 诞 生 以 来 ， 快 速 地 应 用 到 不 同 的 领域 。XML 的 应 用 领域 如 表 15-1 所 示 。 

表 15-1 XML 应 用 领域 


在 XML 诞生 之 前 , 一 些 比较 专业 的 领域 的 信息 无 法 用 一 般 的 标记 语言 表述 出 来 。 现 
在 可 以 使 用 XML 来 表述 这 些 信息 ， 如 CML (Chemical Markup Language) 就 是 使 用 
XML 制定 的 描述 化 学 专业 的 语言 
通常 ， 计 算 机 采用 的 是 二 进 制 存储 数据 ， 如 果 这 些 存储 的 数据 从 一 个 计算 机 传递 到 
另外 一 个 计算 机 上 ， 需 要 对 这 些 二 进 制 数据 进行 读 取 ， 如 果 没 有 专业 的 软件 或 者 数 
据 部 分 损坏 ， 就 不 能 读 取 这 些 信息 。XML 可 以 完全 以 文本 格式 编写 ,会 更 加 方便 地 
进行 文件 的 读 取 
XML 是 一 种 文本 格式 ， 非 常 便于 查看 和 翻译 ， 也 没有 文件 格式 的 版 权 问题 ， 如 使 用 
Word 保存 的 信息 ， 就 必须 有 Word 的 使 用 权 。XML 是 一 种 公开 的 格式 , 没有 版 权 的 
问题 ， 可 以 作为 不 同 计算 机 进行 交换 的 格式 

XML 不 但 可 以 自 定义 标记 ， 还 可 以 对 这 些 自 定义 标记 的 结构 进行 设 定 


定义 专业 领域 的 标记 语言 


通用 的 数据 格式 


数据 交换 容易 


数据 结构 化 


下 面 创建 一 个 简单 的 XML 文件 , 从 感性 上 了 解 XML 文档 的 概念 。 打开 记事 本 , 输入 下 列 内 容 : 
案例 15-1 


<?xml Version="1.0" encoding="utf-8"?> 
< 计算 机 > 

< 名 称 > 联 想 </ 名 称 > 

< 价格 >3264 元 </ 价 格 > 

< 出 厂 日 期 >2007 年 10 月 1 号 </ 出 厂 日 期 > 

</ 计 算 机 > 


将 上 述 代码 保存 ， 名 称 为 Example.xml。 在 保存 该 文件 时 应 使 用 UTF-8 的 字符 集 ， 也 可 以 在 另 
存 为 时 设置 ， 如 图 15-1 所 示 。 
保存 完毕 后 ， 双 击 Example.xml 文件 ， 会 显示 如 图 15-2 所 示 的 执行 结果 。 
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图 15-1 保存 XML 文件 图 15-2 XML 运行 结果 


综 上 所 述 ，XML 是 一 种 描述 数据 和 数据 结构 的 语言 ， 可 以 保存 在 任何 可 以 存储 文本 的 文档 中 。 
XML 具有 自 描述 性 、 内 容 和 显示 相 分 离 、 可 扩张 、 独 立 于 平台 等 特点 。 


15.1.2 XML 优点 


我 们 知道 , XML 是 一 种 标记 语言 。 标 记 语言 包括 文本 、 图 片 和 其 他 多 媒体 链接 ， 可 以 链接 到 当 
前 文档 的 其 他 部 分 ， 或 者 其 他 文档 和 其 他 对 象 等 。 标 记 是 指 编 辑 器 在 待 修 改 的 文档 上 所 做 的 标签 ， 
可 以 用 来 说 明 一 种 事物 或 者 定义 一 个 对 象 。 前 面 提 到 的 SGML， 就 使 用 一 种 标记 语言 的 标准 ， 其 代 
表 是 HTML 语言 。 

HTML 标记 语言 语法 简单 ， 非 常 容 易 上 手 ， 在 初期 没有 任何 定义 文档 外 观 的 相关 方法 ， 仅 用 来 
在 浏览 器 中 显示 网 页 文件 。 而 后 ， 随 着 因特网 的 发 展 ， 人 们 为 了 控制 其 文件 样式 ， 扩 充 了 描述 如 何 
显示 数据 的 标记 。 在 Netscape 与 Microsoft 之 间 的 浏览 器 大 战 后 ，HTML 标准 的 权威 性 遭受 重大 的 
考验 ， 因 为 两 种 浏览 器 支持 不 同 的 标记 。 到 了 HTML 4.0 时 ，W3C 又 恢复 了 其 地 位 ， 同 时 W3C 意 
识 到 HTML 的 如 下 缺点 : 

。 不 能 解决 所 有 解释 数据 的 问题 ， 比 如 影音 文件 或 化 学 公式 、 音 乐 符号 等 其 他 形态 的 内 容 。 

。 效能 问题 。 需 要 下 载 整 份 文件 ， 才 能 开始 对 文件 进行 搜索 。 

e。 扩充 性 、 弹 性 、 易 读 性 均 不 佳 。 

为 了 解决 以 上 问题 ， 专 家 们 使 用 SGML 精简 制作 ， 并 依照 HTML 的 发 展 经 验 ， 产 生出 一 套 使 
用 上 规则 严谨 ， 但 是 简单 的 描述 数据 语言 一 一 XML。 


15.1.3 XML 文档 的 结构 


XML 文件 是 由 标记 及 其 标记 之 间 的 内 容 构成 的 文本 文件 ， 与 HTML 文件 不 同 的 是 ， 这 些 标记 
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可 以 自由 定义 , 其 目的 是 使 XML 文件 能 够 更 好 地 体现 数据 的 结构 和 含义 。 但 是 XML 必须 符合 一 定 
的 语法 规则 ， 这 样 才能 被 XML 解析 器 解析 。XML 文件 可 以 分 为 两 种 类 型 ， 规 范 的 XML 文件 和 有 
效 的 XML 文件 。 符合 W3C 制定 的 基本 语法 规则 的 XML 文件 称 为 规范 的 XML 文件 , 规范 的 XML 
文件 再 符合 额外 的 一 些 约束 就 称 为 有 效 的 XML 文件 。 本 节 首先 介绍 有 效 的 XML 文件 。 

一 个 规范 的 XML 文档 ， 通 常 由 下 列 标记 构成 。 

1. XML 声明 

一 个 规范 的 XML 文档 通常 以 XML 声明 作为 开始 的 第 一 行 。 该 行 的 前 面 不 允许 出 现 空白 、 其 他 
的 处 理 指令 或 注释 。XML 声明 是 以 “<?xml” 开 始 ， 以 “?>” 结 束 。XMEL 声明 的 基本 格式 如 下 
所 示 : 


<?xml version="1.0" encoding="utf-8" standalone="no" ?> 


在 上 述 代 码 中 ， 代 码 “version="1.0"” 表 示 XML 声明 的 版 本 属性 ， 即 创建 的 XML 文件 所 属 的 
版 本 。 一 个 简单 的 XML 文件 可 以 只 包含 XML 声明 。 目 前 该 属性 的 值 只 允许 取 1.0， 因 为 XML 的 
下 一 个 版 本 还 没有 产生 。 如 果 在 XML 声明 中 , 设置 其 他 的 版 本 信息 , 在 运行 XML 时 就 会 出 现 错误 。 

代码 “encoding="utf-8"” 表 示 XML 文件 采用 utf-8 字符 集 进行 编码 。 如 果 在 XML 文件 中 ， 没 
有 指定 该 XML 的 编码 形式 ， 则 其 默认 字符 集 为 utf-8。 如 果 使 用 utf-8 字符 集 ， 就 可 以 在 XML 文件 
中 使 用 中 文 、 英 文 、 日 文 等 各 种 语言 ， XML 解析 器 会 识别 这 些 标记 并 正确 解析 标记 中 的 内 容 。 如 果 
XML 文件 使 用 了 utf-8 字符 集 ， 则 XML 文件 在 保存 时 必须 使 用 utf-8 的 编码 来 保存 。 

通常 ， 也 可 以 使 用 其 他 的 字符 集 来 显示 中 文 和 ASCII 字符 ， 如 GB2312。 其 形式 如 下 所 示 : 


<?Xxml version="1.0" encoding="GB2312"?> 


当 XML 文件 采用 GB2312 编码 时 , 该 文件 必须 使 用 ANSI 编码 保存 。 这 里 可 以 把 GB2312 换 成 
另外 一 种 编码 ISO-8859-1。 

代码 “standalone="no"” 表 示 XML 文件 是 否 是 完全 自 包含 的 ， 即 没有 引用 外 部 实体 。 该 属性 可 
以 取 值 yes 或 no。yes 表示 XML 文件 具有 外 部 实体 ，no 表示 没有 引用 ， 其 默认 值 为 no。 

2. XML 标记 

XML 标记 分 为 非 空 标记 和 空 标记 两 种 。 非 空 标 记 必 须 由 “开始 标记 ”与 “结束 标记 ”组 成 ,“ 开 
始 标 记 ” 与 “结束 标记 ”之 间 是 该 标记 所 具有 的 名 称 。 开 始 标记 以 “<” 标 记 开 始 ， 以 “>” 结 束 ， 
“<” 标 记 与 “>” 标 记 之 间 是 标记 的 名 称 和 属性 列表 。 结 束 标 记 以 “</” 标 记 开 始 ， 以 “>” 标 记 结 
束 ,“</” 标 记 与 “>” 标 记 之 间 是 标记 的 名 称 。 需 要 注意 的 是 ， 在 标记 “</” 和 标记 名 称 之 间 不 要 
有 空格 ， 人 允许“>” 的 前 面 有 空格 和 换行 。 

在 “开始 标记 ”和 “结束 标记 ”之 间 是 该 标记 的 内 容 ， 以 下 是 一 个 正确 的 非 空 标记 : 

< 性 别 > 

男 

</ 性 别 > 

所 谓 空 标 记 就 是 不 包含 任何 内 容 的 标记 。 由 于 空 标 记 不 包含 任何 内 容 ， 所 有 空 标 记 不 需要 开始 
标记 和 结束 标记 。 空 标记 以 “<” 标 记 开 始 ， 以 “/>” 标 记 结 束 ， 二 者 之 间 包 含 空 标记 的 名 称 。 其 使 
用 形式 如 下 所 示 : 

< 电话 号 码 /> 


无 论 是 空 标 记 还 是 非 空 标记 ,其 标记 名 称 必须 满足 一 定 的 规则 ， 其 规则 如 下 : 名 称 可 以 由 字母 、 
数字 、 下 划 线 、 点 (“.”) 和 连接 字符 (“-”) 组 成 ， 但 必须 以 字母 和 下 划 线 开头 。 如 果 XML 文件 的 
字符 集 采 用 的 是 utf-8 编码 ， 则 字母 不 仅 可 以 包含 通常 的 拉丁 字母 ,也 可 以 包含 汉字 等 多 种 语言 。 标 
记名 称 区 分 大 小 写 ， 如 <name> 刘 红 霞 <name> 和 <NAME> 刘 红 霞 <NAME> 是 完全 不 同 的 标记 。 

标记 中 还 有 一 个 特殊 的 标记 为 根 标记 。 XML 文件 必须 有 且 只 有 一 个 根 标记 , 其 他 的 标记 必须 包 
含 在 根 标 记 中 。XML 文件 的 标记 必须 形成 树 状 结构 。 

标记 的 使 用 示例 如 下 所 示 : 

<?xml version="1.0" encoding="utf-8"?> 

<book> 

<name>java 高 级 编程 </name> 
<author> 李 明 伟 </author> 
<name>XML 入 门 </name> 


<author> 刘 海松 </author> 
</book> 


3. XML 属性 

和 HTML 中 标记 的 使 用 一 样 ，XML 标记 在 开始 标记 处 可 以 有 属性 。 属 性 通常 包含 一 些 关 于 标 
记 的 额外 信息 ， 实 际 上 XML 语法 更 倾向 实用 性 ， 因 为 任何 数据 ， 只 要 它 能 通过 标记 来 构造 ， 就 可 
以 用 以 属性 为 中 心 的 方法 来 构造 ， 反 之 亦 然 。 

属性 必须 由 名 字 和 值 组 成 ， 且 必须 在 标记 的 开始 标记 中 声明 ， 并 用 “=” 赋 了 予 属性 的 值 。 其 完 
整 语法 如 下 : 

非 空 标记 ”< 标记 名 属性 名 =" 属 性 值 ”属性 名 =" 属 性 值 "….>…</ 标 记名 > 

空 标记 。 < 标记 名 属性 名 三 "属性 值 ”属性 名 三 "属性 值 "…./> 

使 用 属性 来 描述 标记 的 特征 ， 需 遵守 以 下 规则 : 属性 名 的 命名 规则 和 标记 的 命名 规则 相同 ， 可 
以 由 字母 、 数 字 、 中 文 及 下 划 线 组 成 ， 但 必须 以 字母 、 中 文 或 下 划 线 开头 ， 例 如 : sex、_sex 或 性 别 
都 是 正确 的 。 属 性 名 区 分 大 小 写 ， 例 如 : Sex 和 sex 虽然 描述 的 都 是 性 别 ， 但 在 编译 时 是 两 个 不 同 
的 属性 。 属 性 值 必须 使 用 单 引号 或 双 引 号 ， 例 如 : “male”、“male” 描 述 的 是 相同 的 属性 值 。 如 果 属 
性 值 中 要 使 用 左 尖 括 号 “<” 右 尖 括号 “>” 连接 符号 “&” 单 引号 “'” 或 双 引 号 “"”， 必 须 使 用 
字符 引用 或 实体 引用 。 属 性 的 使 用 示例 如 下 所 示 : 

<?xml] version="]1.0" ?> 

< 学 生 > 

< 学 生 名 称 性 别 =" 男 ”年龄 ="20" 身高 ="175cm” 体重 ="60kg"> 陈 卫 国 </ 学 生 名 称 > 

< 职务 名 称 =" 学 习 委员 "” /> 

</ 学 生 > 

4. XML 注释 

XML 文档 应 该 便于 人 们 阅读 ， 尽 管 XML 文档 解析 器 通常 会 忽略 注释 ， 但 位 置 适当 且 有 意义 的 
注释 可 以 大 大 增加 文档 的 可 读 性 和 清晰 性 ， 所 以 XML 文档 中 不 用 于 描述 数据 的 内 容 都 可 以 包含 在 
注释 中 ， 注 释 以 “<!--” 开 始 ， 以 “-->” 结 束 ， 两 界限 符 之 间 可 以 放任 何 想 输 入 的 字符 。 注 释 语法 
定义 如 下 所 示 : 
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<!-- 注释 内 容 --> 

其 使 用 示例 如 下 所 示 : 

<?xml Version ="1.0" encoding="utf-8"?> 
< 这 是 一 个 注释 一 >。 

< 教程 > 


< 教程 名 称 >XML 基础 入 门 </ 教 程 名 称 > 
</ 教 程 > 


15.1.4 ”命名 空间 


XML 标记 语言 允许 自 定义 标记 ， 那 么 当 在 一 个 XML 文件 中 或 不 同 的 XML 文件 中 定义 了 名 字 
相同 ， 但 内 容 不 同 的 标记 时 ， 如 果 要 区 分 这 些 标记 ， 就 要 使 用 命名 空间 。 在 介绍 命名 空间 之 前 ， 先 
看 一 个 简单 的 XML 文档 。 

<?xml version="1.0" encoding="utf-8" ?> 

< 学 生 名 单 > 

< 张强 >1980 年 出 生 ， 大 学 毕业 </ 张 强 > 

< 马 红 波 >1972 年 出 生 ， 大 学 毕业 </ 马 红 波 > 

< 张强 >1982 年 出 生 ， 大 学 毕业 </ 张 强 > 

</ 学 生 名 单 > 


在 上 述 代码 中 ， 有 两 个 标记 名 称 相同 的 标记 “张强 ” 那么 在 XML 解析 器 解析 这 个 XML 文件 
时 ， 只 能 解析 出 一 个 标记 ， 如 果 想 解析 出 其 中 的 任何 一 个 标记 ， 就 必须 在 XML 文件 中 使 用 命名 
空间 。 

1. 命名 空间 的 定义 

一 个 XML 命名 空间 是 一 个 命名 的 汇集 ， 它 由 URI (统一 资源 标识 符 ) 确定 ， 在 XML 文件 中 作 
为 标记 类 型 和 属性 名 使 用 。 

XML 命名 空间 表示 XML 名 称 的 使 用 范围 , 因为 XML 可 自 定义 标记 标签 , 所 以 在 相同 的 XML 
应 用 间 XML 名 称 重 名 的 机 会 是 很 大 的 。 如 果 没 有 一 种 方法 来 区 分 这 些 应 用 的 名 称 ， 就 会 造成 混乱 。 
XML 命名 空间 就 是 为 了 解决 这 个 问题 而 设计 的 。 

命名 空间 是 XML 文档 的 基本 成 分 ， 是 一 组 保持 唯一 性 的 名 称 。 例 如 ， 可 以 将 一 个 班 的 学 生 姓 
名 作为 一 个 命名 空间 ， 也 可 以 把 一 组 学 生 的 姓名 作为 一 个 命名 空间 。 命 名 空间 就 是 在 逻辑 上 相关 的 
任何 一 组 名 称 , 而且 每 一 个 名 称 都 必须 唯一 。 通过 XML 命名 空间 ,可 以 区 分 来 自 不 同 XML 应 用 的 
具有 相同 名 称 的 标记 和 属性 ， 可 以 将 来 自 单一 XML 应 用 的 相关 标记 和 属性 集合 在 一 起 ， 以 方便 软 
件 识 别 和 处 理 。 

命名 空间 通过 使 用 声明 命名 空间 来 建立 ， 分 为 有 前 绥 命 名 空间 和 无 前 缀 命名 空间 。 

声明 有 前 绥 的 命名 空间 的 语法 格式 如 下 所 示 : 


xmlns: 前 组 = 命名 空间 的 名 字 
例如 : 


xmlns: person="43466421435879"™ 


无 前 级 的 命名 空间 的 声明 语法 格式 如 下 所 示 : 
xmlns= 命 名 空间 的 名 字 
例如 : 


Xmlns=www.tup .com 


在 声明 命名 空间 时 ， 无 论 何 种 命名 空间 的 类 型 ,“xmlns” 与 “: ” 以 及 “: ”与 命名 空间 的 前 缀 
之 间 不 要 有 空格 。 如 果 XML 文件 中 ， 两 个 标记 的 名 称 相同 并 且 命 名 空间 相同 ， 也 就 是 说 ， 对 于 有 
前 缀 的 命名 空间 ， 如 果 两 个 命名 空间 的 名 字 不 相同 ， 即 使 它们 的 前 缀 相同 ， 也 是 不 同 的 命名 空间 ， 
如 果 两 个 命名 空间 相同 ， 即 使 它们 的 前 绥 不 相同 ， 也 是 相同 的 命名 空间 。 命 名 空间 的 前 缀 仅仅 为 了 
方便 地 引用 命名 空间 。 下 列 声明 分 别 声明 了 三 种 不 同类 型 的 命名 空间 : 

xmlns:a="henan™ 


xmlns:a="Henan™ 
xmlns:b="hebei" 


命名 空间 是 区 分 大 小 写 的 ，henan 和 Henan 是 不 同 的 名 称 。 对 于 一 个 标记 的 名 称 的 声明 必须 放 
到 标记 的 “开始 标记 ”中 ， 而 且 命名 空间 的 声明 必须 放 在 开始 标记 中 标记 名 字 的 后 面 ， 如 : 
< 张强 Xmlns :jg="henan"> 


1980 年 出 生 ， 大 学 毕业 
</ 张 强 > 


2. 命名 空间 的 作用 域 
如 果 一 个 标记 使 用 了 命名 空间 声明 ， 那 么 该 命名 空间 的 作用 域 就 是 该 标记 及 其 所 有 的 子孙 
标记 。 
如 果 一 个 标记 中 声明 的 是 具有 前 缀 的 命名 空间 ， 那 么 该 标记 及 其 子孙 标记 都 隶属 于 该 命名 空 
间 ， 则 必须 通过 命名 空间 的 前 组 来 引用 这 个 命名 空间 ， 使 得 该 标记 隶属 于 这 个 命名 空间 。 一 个 标记 
可 以 在 它 的 开始 标记 的 前 面 和 结束 标记 的 名 字 前 添加 命名 空间 的 前 绥 和 冒号 来 引用 该 命名 空间 ， 用 
来 表明 该 标记 的 命名 空间 。 
其 使 用 示例 如 下 所 示 : 
<?xml Version="1.0" encoding="utf-8" ?> 
< 三 班 学 生 名 单 xmlns:pl="www.dlrin.com/company"> 
<p1: 张 水 仙 > 
1983 年 出 生 ， 大 学 毕业 
<p1 :小 张 水 仙 > 在 小 学 读书 <p1 : 小 张 水 仙 > 
</pl: 张 水 仙 > 
< 刘海 防 > 
1978 年 出 生 ， 大 学 毕业 
</ 刘 海防 > 
</ 三 班 学 生 名 单 > 


在 上 述 XML 文件 中 ,“ 张 水 仙 ” 和 “小 张 水 仙 ” 素 属于 同一 个 命名 空间 “p1”。 
如 果 一 个 标记 中 声明 了 无 前 绥 的 命名 空间 ， 那 么 该 标记 及 其 子孙 标记 都 默认 地 隶属 于 这 个 命名 
空间 。 其 使 用 示例 如 下 所 示 : 
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<?xml] Version="1.0"” encoding="utf-8" ?> 
<book xmlns="www.itzcn.com"> 
<java>java 基础 教程 </java> 
<jsp>jsp 基础 教程 </jsp> 
</book> 
在 上 述 代 码 中 ，java 标记 和 jsp 标记 的 命名 空间 都 属于 www.itzcn.com。 不 仅 可 以 在 子 标记 的 前 
面 通过 命名 空间 的 前 级 来 引用 父 标记 声明 的 有 前 缀 的 命名 空间 ， 而 且 子 标记 也 可 以 重新 声明 命名 空 
间 。 其 使 用 示例 如 下 所 示 : 
<?xml version="1.0" encoding="utf-8" ?> 
< 三 班 学 生 名 单 xmlns:pl="www.dlrin.com/company"> 
<p1: 张 水 仙 > 
1983 年 出 生 ， 大 学 毕业 
<p1l1: 小 张 水 仙 xmlns=beijing> 在 小 学 读书 <pl :小 张 水 仙 > 
<p1: 小 张 水 仙 xmlns:p2=aAmerica> 在 小 学 读书 <pl :小 张 水 仙 > 
</p1: 张 水 仙 > 
</ 三 班 学 生 名 单 > 


命名 空间 的 目的 是 有 效 地 区 分 名 字 相同 的 标记 。 对 于 命名 空间 的 命名 W3C 推荐 使 用 统一 资源 
标识 符 (URI) 作为 命名 空间 的 名 字 。URI 是 有 一 定 的 语法 、 用 来 标识 资源 的 一 个 字符 串 。 一 个 URI 
可 以 是 一 个 E-mail 地 址 、 一 个 文件 的 绝对 路 径 、 一 个 Internet 主机 的 域名 等 。 


个 .1.5 DTD 


对 于 一 个 格式 良好 的 XML 文档 , 只 能 保证 这 个 文档 的 格式 符合 XML 规范 , 但 是 标记 与 标记 的 
关系 、 标 记 与 属性 间 的 关系 、 属 性 的 取 值 是 否 正确 等 ， 这 些 都 无 法 进行 判断 ， 因 为 这 些 没有 一 个 标 
准 来 衡量 。 对 于 一 个 格式 良好 、 符 合 规范 的 XML 文档， 如 果 仅 仅 是 在 有 限 中 使 用 ， 或 者 作为 数据 
的 传输 ， 那 么 也 能 很 好 地 满足 实际 应 用 。 但 如 果 让 其 他 的 用 户 来 理解 这 个 XML 文档 ， 获 取 和 其 他 
的 应 用 进行 数据 交互 , 那么 就 有 必要 提供 一 种 机 制 , 来 保证 编写 的 XML 文档 和 别人 写 的 XML 文档 
其 结构 是 相同 的 ， 标 记 和 标记 之 间 的 关系 是 正确 的 ， 属 性 的 取 值 也 是 符合 要 求 的 。 

这 种 机 制 就 是 DTD (Document Type Definition， 文档 类 型 定义 )。 一 个 规范 的 XML 文件 如 果 和 
某 个 DTD 文件 相关 联 ， 并 遵守 DTD 文件 规定 的 限制 条 件 ， 就 称 为 有 效 的 XML 文件 。 需 要 注意 的 
是 ，DTD 的 编码 字符 集 需 要 和 关联 的 XML 文件 保持 一 致 。 

1. DTD 的 使 用 及 结构 

在 XML 文档 中 包含 文档 类 型 声明 ， 从 而 达到 当前 文档 和 DTD 关联 的 目的 。 当 进行 有 效 性 验证 
的 XML 处 理 器 读 到 该 指令 时 ， 它 获取 DTD， 并 根据 其 中 定义 的 规则 对 文档 进行 检验 。 在 XML 文 
档 中 ， 使 用 DTD 有 两 种 方式 ， 一 种 是 直接 在 XML 文档 中 定义 DTD， 另 一 种 是 通过 URI 引用 外 部 
的 DTD 文件 ， 或 者 同时 采用 这 两 种 方法 。 

文档 类 型 声明 以 <IDOCTYPE 开始 ， 以 ]> 结 束 。 通 常 将 开始 和 结束 放 在 不 同 的 行 上 ， 但 断 行 和 
多 余 的 空格 并 不 重要 。 文 档 类 型 定义 的 内 部 子 集 ( 即 内 部 DID) 是 定义 在 XML 文档 内 部 的 ， 其 语 
法 格式 如 下 所 示 : 


<!DOCTYPE 根 标记 名 称 [ 


<!ELEMENT 子 标记 名 称 (#PCDATA) > 
2 


。 <!DOCTYPE 关键 词 ， 表 示 定 义 DTD， 且 必须 为 大 写 。 

。 根 标记 名 称 一 个 XML 文档 只 有 一 个 根 标 记 。 如 果 XML 文档 使 用 DTD， 那 么 文档 中 的 根 
标记 名 就 在 内 部 子 集中 指定 。 

。 [<!ELEMENT 子 标记 名 称 (#PCDATA)>] 定义 文档 构成 的 标记 . <!IELEMENT 子 标记 名 称 
(#PCDATA)> 是 标记 类 型 声明 。 

。 > 结束 DTD 的 定义 。 

内 部 DTD 的 使 用 示例 如 下 所 示 : 


<?xml] version ="1.0" encoding="utf-8" standalone="yes" ?> 
<!DOCTYPE 员工 [ 
<!ELEMENT 姓名 (#PCDATA)> 
<!ELEMENT 性 别 (#PCDRTR) > 
<!ELEMENT 出 生日 期 (#PCDATA)> 
]> 
< 员工 > 
< 姓名 > 李 丽 </ 姓 名 > 
< 性 别 > 女 </ 性 别 > 
< 出 生日 期 >1978 .2.5</ 出 生日 期 > 
</ 员 工 > 


在 上 述 代码 中 ， 定 义 了 根 标记 名 称 为 员工 ， 该 标记 具有 三 个 子 标记 : 姓名 、 性 别 和 出 生日 期 。 
XML 文档 通过 一 个 外 部 的 URL 连接 到 DTD， 称 为 外 部 子 集 ( 即 外 部 DTD)。 外 部 子 集 在 物理 


上 位 于 另 一 个 文件 中 ,扩展 名 为 .dtd。 它 可 以 供 多 个 XML 文档 使 用 ， 就 像 同一 结构 可 以 写 出 多 个 不 
同 内 容 的 文档 一 样 ， 多 个 XML 文档 是 因为 引用 了 同一 个 外 部 DTD， 所 以 它们 的 结构 大 致 相同 。 在 
XML 文件 中 引用 创建 好 的 外 部 DTD 的 引用 语句 ， 必 须 位 于 XML 文件 的 文档 类 型 定义 部 分 ， 语 法 
格式 声明 如 下 所 示 : 


或 


<!DOCTYPE 根 标记 名 称 SYSTEM "DTD-URL"> 


<!DOCTYPE 根 标记 名 称 PUBLIC "DTD-name" "DTD-URL"> 


。 <!DOCTYPE 关键 字 ， 表 示 引 用 外 部 DID。 

。 根 标记 名 称 ”在 外 部 DTD 中 定义 的 根 标记 。 

。 SYSTEM 关键 字 ， 指 该 外 部 DTD 中 文件 是 私有 的 ， 即 由 用 户 创建 但 并 没有 公开 发 行 ， 只 
是 个 人 或 几 个 合作 者 之 间 使 用 。 

。 PUBLIC 关键 字 ， 指 该 外 部 DTD 文件 是 公有 的 ， 用 PUBLIC 的 DTD 都 有 一 个 逻辑 名 称 
DTD-name,， 必须 在 调用 时 指明 这 个 远 辑 名 称 . 使 用 PUBLIC 关键 字 通 常 表示 DTD 的 使 用 范 
围 比较 大 (例如 在 HTML 文档 中 也 可 以 使 用 DTD )。 

。 DTD-URL 通过 URL 将 外 部 DID 引用 到 XML 文档 中 。 例 如 有 一 个 名 为 “bookrule.dtd” 
的 外 部 DTD 文件 放 在 某 出 版 社 的 URL 为 www.bookpublish.com 的 地 方 ， 那 么 在 XML 文档 
中 的 引用 是 : <IDOCTYPE Book SYSTEM "http://www.bookpublish.com/bookrule.dtd">。 
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外 部 DTD 的 使 用 示例 如 下 所 示 : 


<?xml] version="1.0" encoding="utf-8" ?> 
<!ELEMENT 联系 方式 (姓名 , 电子 邮件 *) > 
<!ELEMENT 姓名 (#PCDATA)> 

<!ELEMENT 电子 邮件 (#PCDATA)> 


在 下 面 的 XML 文档 中 引入 该 DTD 文件 ， 如 下 所 示 : 


<?xml version="1.0" standalone="no"?> 
<!DOCTYPE 联系 方式 SYSTEM "external.dtd"> 
< 联系 方式 > 

< 姓名 >Jims</ 姓 名 > 

< 电子 邮件 >Jims@163 .com</ 电 子 邮 件 > 

< 电子 邮件 >Jims@21cn.com</ 电 子 邮件 > 
</ 联 系 方式 > 


在 XML 文档 中 ， 也 可 以 同时 采用 上 面 的 两 种 方式 。 

DTD 的 结构 一 般 由 标记 类 型 声明 、 属 性 声明 、 实 体 声明 、 记 号 声明 等 构成 。 一 个 典型 的 文档 类 
型 定义 文件 会 把 将 来 要 创建 的 XML 文档 的 标记 结构 、 属 性 类 型 、 实 体 引 用 等 预先 进行 定义 。 

2. DTD 的 标记 声明 

上 节 文 档 类 型 声明 中 的 每 一 项 都 是 标记 声明 , 定义 了 每 个 标记 的 约束 。 标记 声明 的 格式 如 下 所 示 : 


<!ELEMENT element name (content model)> 


有 效 文档 中 使 用 的 每 个 标记 都 必须 在 文档 的 DTD 中 用 标记 声明 进行 声明 。element_name 可 以 
是 任何 合法 的 XML 名 称 , content_model( 内 容 模型 ) 指 定 标记 可 以 或 必须 包含 的 子 标记 以 及 子 标记 的 
顺序 。 下 面具 体 介绍 内 容 模型 的 内 容 。 

(1) # 才 CDATA: 规定 标记 只 包含 字符 数据 。 下 面 的 声明 指出 一 个 name 标记 可 以 包含 文本 ， 但 
不 能 划分 为 独立 的 area_code、number 和 extension 标记 。 


<!ELEMENT name (#PCDATA)> 

(2) 子 标记 : 可 指明 标记 的 子 标记 。 下 面 的 声明 表示 name 标记 必须 包含 且 只 包含 一 个 desc 
标记 。 

<!IELEMENT name (desc)> 

也 可 用 逗号 作为 分 隔 符 ， 指 明 多 个 子 标 记 ， 并 且 子 标记 出 现 的 次 序 必 须 按 定义 时 的 顺序 ， 如 下 
所 示 : 

<!ELEMENT name (id,desc)> 

name 标记 的 id 子 标记 必须 在 desc 子 标记 前 面 , 否则 验证 会 出 错 , 该 文档 不 是 一 个 有 效 的 XML 
文档 。 

下 面 这 个 文档 是 有 效 的 。 


<name> 
<id>1</id> 


下 面 这 个 文档 是 无 效 的 ， 顺 序 颠 倒 了 。 


下 面 的 文档 也 是 无 效 的 ， 有 多 余 的 标记 。 


可 以 通过 正则 表达 式 来 规定 子 标记 的 个 数 。 其 常用 符号 如 表 15-2 所 示 。 
表 15-2 元 素 指 示 符 


允许 一 个 或 多 个 该 标记 
允许 零 个 或 多 个 该 标记 
允许 零 个 或 一 个 该 标记 


下 面 利用 这 些 符号 规定 id 子 标记 必须 出 现 ， 且 只 能 出 现 一 次 ， 而 desc 子 标记 可 选 。 


根据 上 面 的 声明 ， 下 面 的 name 标记 都 是 有 效 的 。 


可 选项 (|) 是 一 个 参数 列表 ， 每 个 参数 间 用 “|” 分 隔 ， 代 表 能 且 只 能 选 一 个 子 元 素 。 
<IEIEMENT choice (good | bad> 


上 例 的 choice 元 素 可 选 一 个 good 子 元 素 ， 或 bad 子 元 素 ， 且 只 能 从 中 选择 一 个 。 可 选 的 参数 
列 可 以 多 项 ， 不 限于 两 项 ， 如 下 所 示 : 


巴 浊 


项 


卫 
= 
五 
和 
x 
三 


<!ELEMENT choice (one | two | three | four)> 
可 用 小 括号 把 选项 括 起 来 ， 以 表达 更 丰富 的 意思 ， 如 想 表示 choice 元 素 必 须 包 含 一 个 good 子 
元 素 ， 并 且 必 须 包 含 ok 子 元 素 或 bad 子 元 素 的 一 个 。 

<!ELEMENT choice (good, (oklbad))> 

(3) 混合 内 容 : 在 一 些 文档 中 ， 一 个 元 素 可 能 既 包 含 子 元 素 ， 也 包含 字符 串 ， 这 些 内 容 叫做 混 
合 内 容 。 可 用 以 下 方式 表示 : 


<!EMEMENT description (#PCDATA | term)* )> 


该 声明 表示 description 元 素 可 包含 已 析 的 字符 串 和 term 子 元 素 ， 且 人 允许 出 现 零 次 或 多 次 , 如 下 
所 示 : 


<description> 
this is a <term>dtd</term> test. 
</description> 


#PCDATA 必须 在 第 一 位 ， 可 选 的 子 元 素 可 任意 多 项 。 

(4) 空 元 素 : 某 些 元 素 不 包含 任何 内 容 ， 称 之 为 空 元 素 ， 写 成 以 放 结 束 的 独立 标签 。 

<!ELEMENT image EMPTY> 

示例 : 

<image src="http://www.xml.com/dtd.jpg" /> 

(5) ANY: 允许 元 素 内 包含 任意 内 容 。 该 选项 在 DTD 测试 时 很 有 用 ， 在 生产 系统 中 尽量 不 要 
使 用 。 

<!ELEMENT page ANY> 

3. DTD 的 属性 声明 


一 个 有 效 的 XML 文档 , 必须 对 元 素 的 属性 进行 声明 。 使 用 ATTLIST 声明 来 完成 ,一 个 ATTLIST 
可 以 为 一 个 元 素 类 型 声明 多 个 属性 。 
<!RATTLIST image src CDATA #REQUIRED> 
上 例 声明 image 元 素 必须 有 一 个 src 属性 ， 该 属性 的 值 是 字符 数据 。 可 用 ATTLIST 声明 为 一 个 
元 素 声明 多 个 属性 ， 如 下 所 示 : 
<!ATTLIST image src CDATA #REQUIRED 
width CDATA #REQUIRED 


height CDATA #REQUIRED 
alt CDATA #IMPLIED 
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2 
上 述 声明 指出 src、width、height 属性 是 必需 的 ，alt 属性 是 可 选 的 。 


(1) 属性 类 型 
常用 的 属性 类 型 如 表 15-3 所 示 。 


表 15-3 属性 类 型 表 


属性 类 型 BN 
CDATA 字符 数据 ， 即 没有 标记 的 文本 
枚 举 可 选择 的 可 能 值 列表 
ID 不 被 文档 中 任何 其 他 ID 类 型 属性 共享 的 唯一 的 名 称 
IDREF 文档 中 元 素 的 ID 类 型 属性 的 值 
IDREFS 由 空格 分 隔 的 元 素 的 多 个 人 D 
ENTITY 在 DTD 中 声明 的 实体 的 名 称 
ENTITIES 在 DID 中 声明 的 由 空格 分 隔 的 多 个 实体 的 名 称 
NMTOKEN XML 名 称 记号 
NMTOKENS 由 空格 分 隔 的 多 个 XML 名 称 记号 
NOTATION 在 DTD 中 声明 的 记 法 的 名 称 


CDATA 类 型 属性 值 可 包含 任意 文本 字符 串 .DTD 不 能 指定 属性 为 一 个 整数 或 一 个 日 期 , Schema 
能 提供 更 为 强大 的 数据 类 型 。 

NMTOKEN 类 型 属性 值 是 一 个 XML 名 称 记号 。XML 名 称 记号 与 XML 名 称 类 似 ， 但 XML 名 
称 记 号 允许 所 有 的 字符 作为 名 称 的 开始 字符 ， 而 XML 名 称 的 第 一 个 字母 必须 是 字母 、 表 意 字符 和 
下 划 线 ， 因 此 10、.bashre 是 合法 的 XML 名 称 标 记 ， 但 不 是 合法 的 XML 名 称 。 每 个 XML 名 称 都 
是 一 个 XML 名 称 标 记 ， 然 而 XML 名 称 标记 不 全 是 XML 名 称 。 如 果 属 性 包含 1990、2005 之 类 的 
整数 ， 则 应 该 指定 其 类 型 为 NMTOKEN， 如 下 所 示 : 


<!ELEMENT person birthday NMTOKEN #REQUIRED> 

NMTOKENS 类 型 属性 包含 一 个 或 多 个 用 空白 分 隔 的 XML 名称 记 号 ， 如 下 所 示 : 

<person dates="02-01-2005 03-01-2005 05-01-2005">person</person> 

对 应 的 声明 应 如 下 所 示 : 

<!ATTLIST person dates NMTOKENS #REQUIRED> 

男 一 方面 , 对 01/02/2005 这 样 的 形式 不 能 使 用 该 声明 , 因为 其 中 的 正 斜 杠 不 是 合法 的 名 称 字符 。 
枚 举 声明 ， 枚 举 不 用 关键 字 。 直 接 列 举 所 有 的 值 ， 中 间 用 竖 线 分 隔 。 如 : 


<!ATTLIST date month (January1Eebruary1March | April | May | June | JulylAugust | 
September | October | November | December) #REQUIRED> 


针对 上 述 声 明 ，date 元 素 的 month 属性 可 选 12 个 月 份 中 的 一 个 。 

ID 类 型 的 属性 必须 包含 一 个 XML 名 称 ， 而 且 该 名 称 在 文档 中 是 独一无二 的 。ID 属性 可 为 元 素 
分 配 一 个 唯一 的 标识 符 。 

<!RTTLIST name card id ID #REQUIRED> 
由 于 数字 不 是 合法 的 XML 名 称 ， 所 以 编号 不 能 以 数字 开头 ， 解 决 办 法 是 在 前 面 加 下 划 线 或 
字母 。 

IDREF 类 型 的 属性 指向 文档 中 某 元 素 的 ID 类 型 的 属性 ， 因此, 它 必 须 是 一 个 XML 名 称 , 它 的 
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作用 是 当 简 单 的 包含 关系 不 能 满足 要 求 时 ， 在 元 素 间 建立 多 对 多 关系 ， 如 下 所 示 : 


<project project id="p1"> 
<goal>deploy linux</goal> 
<team member person card id="c123"> 
</project> 
<person card id="c123"> 
<name>linuxsir</name> 
<assignment project project id="pl"> 
</person> 


project 元 素 的 project_id 属性 和 person 元 素 的 card_id 属性 应 该 是 ID 类 型 。team_member 元 素 
的 person_card id 属性 和 assignment 元 素 的 project_project id 属性 是 IDREF 类 型 。 对 应 的 声明 如 下 
所 示 : 


<!ATTLIST person card id ID #REQUIRED> 
<!ATTLIST project project id ID #REQUIRED> 
<!ATTLIST team member person card id IDREF #REQUIRED> 
<!ATTLIST assignment project project id IDREF #REQUIRED> 
IDREF 类 型 的 属性 包含 一 个 XML 名 称 列 表 。 名 称 间 用 空格 间隔 ， 且 每 个 名 称 都 是 文档 中 某 个 元 素 的 ID。 当 某 
个 元 素 需要 引用 多 个 其 他 元 素 时 使 用 该 元 素 ， 如 下 所 示 : 
<!ATTLIST Person card id ID #REQUIRED 
assignment IDREFS #REQUIRED> 
<!ATTLIST project project id ID #REQUIRED 
team IDREFS #REQUIRED> 


对 应 的 文档 可 写成 如 下 所 示 的 形式 : 


<project project id="pl" team="c1l23"> 
<gold>deploy linux</gold> 

</project> 

<person card id="cl23" assignment="pl"> 
<name>Linuxsir</name> 

</person> 


ENTITY 类 型 的 属性 包含 在 DTD 的 其 他 位 置 声明 的 未 析 实 体 名 称 中 , 如 movie 元 素 可 能 有 一 个 
标识 激活 时 播放 mpeg 或 mm 文件 的 实体 属性 。 


<!ATTLIST movie src ENTITY #REQUIRED> 


如 果 DTD 声明 了 一 个 名 为 play 的 未 析 实 体 ， 则 此 movie 元 素 可 用 于 在 XML 文档 中 嵌入 视频 
文件 。 
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<movie src="play" /> 
ENTITIES 类 型 的 属性 包含 在 DTD 的 其 他 位 置 声明 的 多 个 未 析 实 体 名 称 中 ， 其 间 用 空白 隔 开 。 
<!ATTLIST slide show slides ENTITIES #REQUIRED> 


如 果 DID 声明 了 未 析 实 体 slidel、slide2、slide3、…， 则 可 使 用 slide_show 元 素 在 XML 文档 
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中 嵌入 幻灯 片 。 
<slide show slides="slidel slide2 slide3" /> 


NOTATION 类 型 的 属性 包含 在 文档 的 DTD 中 声明 的 某 个 记 法 的 名 称 。 该 属性 类 型 较 少 用 。 理 
论 上 , 可 以 使 用 该 属性 使 某 些 特殊 元 素 与 类 型 相关 联 , 下 例 声明 为 不 同 的 图 像 类 型 定义 了 4 个 记 法 ， 。 
然后 规定 每 个 image 元 素 都 必须 从 中 选择 一 种 type 属性 。 

<!NOTRTION gif SYSTEM "image/gif"> 

<!NOTRTION tiff SYSTEM "image/tiff"> 


<!NOTATION jpeg SYSTEM "image/jpeg"> 
<!NOTATION png SYSTEM "image/png"> 


<!ATTLIST image type NOTATION (gif | tiff | jpeg | png) #REQUIRED> 第 
15 
每 个 image 元 素 的 type 属性 值 可 以 为 gif、ti 企 、jpeg 和 png 四 个 值 中 的 一 个 。 该 属性 比 枚 举 类 章 
型 稍 具 优势 ， 因 为 记 法 的 实际 MIME 媒体 类 型 在 理论 上 是 可 用 的 。 由 于 和 斜 杠 在 XML 名 称 中 不 是 一 0 
个 合法 字符 ， 所 以 枚 举 类 型 不 能 指定 image/png 或 image/ipeg 作为 允许 值 。 五 
(2) 属性 默认 值 区 
每 个 AITLIST 声明 除了 要 提供 一 种 数据 类 型 外 , 还 要 声明 属性 的 默认 行为 。 其 常用 的 默认 值 如 三 
表 15-4 所 示 。 
表 15-4 ”属性 默认 值 
属性 默认 值 含义 
#REQUIRED 元 素 的 每 个 实例 必须 包含 该 属性 
#IMPLIED 元 素 实例 可 以 选择 是 否 包含 该 属性 
证 IXED+ 默 认 值 ee 如 果 元 素 中 不 包含 该 属性 的 属性 值 ， 则 解析 器 会 将 默认 值 作为 
和 如 果 元 素 中 不 包含 该 属性 的 属性 值 ， 解 析 器 会 将 默认 值 作为 属性 值 ， 否 则 该 属性 可 以 有 其 
默认 值 
他 属性 值 
其 常用 示例 如 下 所 示 : 


<!ATTLIST person name NMTOKEN "linuxsir™ 
如 果 没 有 显 式 指明 person 元 素 的 name 属性 ， 则 该 值 为 linuxsir。 
15.1.6 ”相关 技术 
XML 的 应 用 领域 越 来 越 多 ， 同 时 也 形成 了 与 XML 相关 的 多 种 技术 。 其 常用 技术 如 表 15-5 


表 15-5 XML 相 关 技 术 


技 术 说 阴 
可 扩展 的 HTML: XHTML 是 HTML 4.01， 它 是 在 XML 中 的 再 生成 产物 。XHTML 1.0 是 HTML 的 
最 新 版 本 
CSS 层叠 样 式 表 : 可 以 将 CSS 格式 应 用 到 XML 文档 中 来 提供 数据 
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续 表 
技 术 说 了 明 


可 扩展 的 格式 表 语 言 : XSL 包括 三 部 分 ，XML 文档 转换 〈 重 命名 为 XSLT)， 一 个 格式 匹配 语法 〈 重 
命名 为 XPath)， 以 及 一 个 格式 化 对 象 解释 


XSLT XML 转换 : XSLT 比 CSS 功能 强大 得 多 。 它 可 以 用 来 将 XML 文件 转换 成 多 种 不 同 的 输出 格式 


XML 格式 匹配 :XPath 是 用 来 对 一 个 XML 文档 的 一 部 分 进行 寻 址 的 语言 ,XPath 在 设计 时 是 让 XSLT 
和 XPointer 二 者 都 使 用 的 


XLink XML 连接 语言 XML 连接 语言 (XLink) 允 许 将 元 素 插入 XML 文档 中 ， 以 创建 XML 资源 之 间 的 链接 
XPointer | XML 指针 语言 XML 指针 语言 (XPointer) 支 持 对 XML 文档 内 部 结构 的 寻 址 , 例如 元 素 、 属性 和 内 容 


DTD 文档 类 型 定义 : 一 个 DTD 可 以 用 来 定义 一 个 XML 文档 的 合法 结构 区 
XML 命名 空间 定义 了 一 种 方法 ， 这 种 方法 通过 将 XML 中 使 用 的 元 素 和 属性 与 URI 引用 联系 起 来 ， 
来 对 元 素 和 属性 进行 定义 


XSD XML 计划 : 计划 是 DTD 的 强 有 力 的 替代 品 。 计 划 是 用 XML 编写 的 ， 它 支持 命名 空间 和 数据 类 型 


XML 的 简化 数据 : XDR 是 XML 计划 的 一 个 简化 版 本 。 对 XDR 的 支持 是 mntemet Explorer 5.0 携带 
XDR 的 ， 而 XML 计划 还 只 是 一 个 工作 蓝本 。 一 旦 XML 计划 的 规范 成 为 W3C 推荐 的 ，Microsoft 就 会 全 
面 支持 它 

DOM DOM 定义 处 理 XML 文档 的 界面 、 属 性 和 方法 

XQL XML 查询 语言 ，XML 查询 语言 支持 查询 功能 ， 从 XML 文档 中 提取 数据 

SAX SAX 是 阅读 和 操作 XML 文档 的 另 一 个 界面 


15.2 在 PHP 中 处 理 XML 


使 用 PHP 技术 ， 可 以 对 一 个 XML 文档 进行 操作 。 其 常见 的 操作 有 获取 数据 、 添 加 数据 、 修 改 
数据 、 删 除数 据 等 。 如 果 要 实现 对 XML 文档 操作 ，PHP 要 实现 相应 的 XML 接口 ， 并 实现 这 些 接 
口中 的 方法 ， 才 能 进行 操作 。 其 常见 的 接口 有 两 个 ， 分 别 为 SAX 和 DOM 接口 。 本 节 将 对 这 两 个 接 
口 进行 分 析 ， 并 使 用 DOM 接口 对 XML 文档 进行 相应 的 操作 。 


15.2.1 解析 方法 比较 


在 PHP 5.0 中 ， 对 XML 进行 解析 ， 可 以 使 用 DOM 库 、SAX 解析 器 和 正则 表达 式 三 种 方式 。 
同样 也 可 以 采用 其 他 的 方式 , 如 从 网 络 上 下 载 PEAR 包 或 者 利用 PHP 本 身 提供 的 SimpleXML 函数 。 

使 用 DOM 接口 处 理 XML 文档 ， 其 机 制 如 下 : 需要 把 整个 XML 文件 加 载 到 内 存 中 去 ， 其 存在 
形式 是 以 一 棵 节点 树 存在 的 ， 当 完成 加 载 后 ， 可 以 对 该 节点 树 中 的 每 一 个 节点 进行 操作 。 换 句 话说 ， 
通过 DOM 树 ， 应 用 程序 可 以 对 XML 文档 进行 随机 访问 。 这 种 访问 方式 给 应 用 程序 的 开发 带 来 了 
很 大 的 灵活 性 ， 它 可 以 任意 地 控制 整个 XML 文档 中 的 内 容 。 执 行 添加 、 删 除 、 修 改 等 操作 。 

然而 ， 由 于 DOM 解析 器 把 整个 XML 文档 转化 成 DOM 树 放 在 了 内 存 中 ， 因 此 ， 当 XML 文档 
比较 大 或 者 文档 结构 比较 复杂 时 ， 对 内 存 的 需求 就 比较 高 。 而 且 ， 对 于 结构 复杂 的 树 的 遍历 也 是 一 
项 比较 耗 时 的 操作 。 所 以 ， DOM 解析 器 对 计算 机 性 能 的 要 求 比较 高 ， 实 现 效率 不 十 分 理想 。 不 过 ， 
由 于 DOM 解析 器 的 树 结构 的 思想 与 XML 文档 的 结构 相 吻 合 , 而且, 通过 DOM 树 机 制 很 容易 实现 


随机 访问 ， 因 此 仍然 可 以 使 用 这 种 方法 。 

读 取 XML 的 另 一 种 方法 是 使 用 XML Simple API(SAX) 解 析 器 .PHP 的 大 多 数 安装 都 包含 SAX 
解析 器 。SAX 解析 器 运行 在 回调 模型 上 。 每 次 打开 或 关闭 一 个 标记 时 , 或 者 每 次 解析 器 看 到 文本 时 ， 
就 用 节点 或 文本 的 信息 回调 用 户 定义 的 函数 。SAX 解析 器 的 优点 是 ， 它 是 真正 轻 量 级 的 。 解 析 器 不 
会 在 内 存 中 长 期 保持 内 容 , 所 以 可 以 用 于 非常 巨大 的 文件 。 缺 点 是 编写 SAX 解析 器 回调 是 件 非常 麻 
烦 的 事情 。 

同 DOM 分 析 器 相 比 ，SAX 解析 器 对 XML 文档 的 处 理 缺 乏 一 定 的 灵活 性 ， 然 而 ， 对 于 那些 只 
需要 访问 XML 文档 中 的 数据 而 不 对 文档 进行 更 改 的 应 用 程序 来 说 ，SAX 解析 器 的 效率 则 更 高 。 由 
于 SAX 解析 器 实现 简单 ， 对 内 存 要 求 比较 低 ， 因 此 实现 效率 比较 高 。 

使 用 正则 表达 式 代 码 读 取 XML 的 问题 是 , 它 并 没有 先进 行 检查 , 确保 XML 的 格式 良好 。 这 意 
味 着 在 读 取 之 前 , 无 法 知道 XML 是 否 格式 良好 。 而且 ， 有 些 格式 正确 的 XML 可 能 与 正则 表达 式 不 
匹配 ， 所 以 日 后 必须 修改 它们 。 从 不 建议 使 用 正则 表达 式 读 取 XML, 但 是 有 时 它 是 兼容 性 最 好 的 方 
式 ， 因 为 正则 表达 式 函 数 总 是 可 用 的 。 不 要 用 正则 表达 式 读 取 直 接 来 自用 户 的 XML， 因为 无 法 控制 
这 类 XML 的 格式 或 结构 。 建 议 选用 DOM 库 或 SAX 解析 器 读 取 来 自用 户 的 XML。 


15.2.2 ”使 用 DOM 接口 


DOM 是 Document Object Model 的 英文 缩写 ， 即 文档 对 象 模型 。W3C 文档 对 象 模型 (DOM) 
可 以 看 作 是 一 个 平台 或 与 语言 无 关 的 〈language-neutral) 界面 ， 它 允许 程序 和 脚本 动态 地 访问 以 及 
更 新 文档 的 内 容 、 结 构 、 脚 本 程序 。 在 这 里 DOM 仅仅 是 一 种 对 某 种 功能 和 结构 的 声明 ， 告 诉 别 的 
对 象 具有 什么 样 的 概念 定义 ， 如 果 遵循 该 定义 ， 那 么 可 以 完成 什么 样 的 功能 。 

简单 地 看 ，DOM 可 以 看 作 是 一 组 API (Application Program Interface， 应 用 程序 接口 )， 它 把 
HTML 文档 、XML 文档 等 看 作 一 个 文档 对 象 ， 里 面 存放 的 是 对 这 些 文档 操作 的 属性 和 方法 的 定义 ， 
若 编程 语言 实现 了 这 些 属性 和 方法 ， 就 可 以 对 文档 对 象 中 的 数据 进行 存 取 ， 并 且 利用 程序 对 数据 进 
行进 一 步 的 处 理 。 

W3C DOM 提供 了 一 组 描述 HTML 及 XML 文件 的 标准 对 象 和 一 个 用 来 访问 和 操作 这 类 文件 的 
标准 界面 。 若 以 面向 对 象 的 思维 来 看 , 可 以 把 HTML 文档 或 XML 文档 看 作 是 一 个 对 象 , 一 个 XML 
文档 对 象 可 以 包含 其 他 的 对 象 ， 如 节点 对 象 。 对 XML 文档 对 象 的 操作 实际 是 对 该 对 象 的 节点 对 象 
的 操作 ， 可 以 对 对 象 进行 修改 等 操作 。 在 DOM 中 有 相应 的 对 象 对 应 实际 上 的 XML 文档 对 象 ， 那 
么 也 可 以 这 样 理解 DOM， 在 DOM 规范 中 提供 了 一 组 对 象 对 文档 结构 的 访问 。 

PHP 语言 实现 了 DOM 接口 ， 这 就 意味 着 可 以 利用 W3C 提供 的 DOM 接口 操作 XML 文档 。 在 
对 XML 文档 操作 之 前 ， 需 要 把 XML 文档 加 载 到 内 存 中 ， 形 成 一 棵 节点 树 。 树 上 的 每 一 个 节点 ， 就 
是 XML 文档 的 每 一 个 标记 和 相应 的 标记 内 容 。 理 解 这 一 点 对 于 后 面 编写 PHP 程序 特别 重要 。 使 用 
DOM 接口 对 XML 进行 操作 ， 在 PHP 中 可 以 使 用 两 个 类 库 ， 一 个 为 DOM Functions 函数 库 ， 另 外 
一 个 为 DOM XML Functions 函数 库 。 第 一 个 函数 库 是 内 置 的 ， 直接 可 以 调用 该 函数 库 中 的 函数 ; 第 
二 个 函数 库 需 要 在 php.ini 文件 中 引入 。 这 里 主要 介绍 使 用 内 置 函数 库 对 XML 文件 解析 。 

对 XML 文件 的 操作 , 常见 的 有 人 遍历、 删除 、 修改、 添加 等 操作 。 下 面 结合 对 XML 文件 的 操作 ， 
介绍 DOM Functions 函数 库 中 的 函数 。 
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1. 遍历 节点 
前 面 介 绍 过 ，XML 文件 是 用 来 存储 数据 、 传 输 数据 的 。 编 写 一 个 PHP 程序 从 XML 文件 中 获 
取 数 据 并 输出 ， 需 要 使 用 到 DOM Functions 函数 库 中 的 函数 。 这 些 常用 的 函数 如 表 15-6 所 示 。 
表 15-6 ”DOM Functions 函 数 


名 称 说 明 
DOMDocumentO 创建 一 个 DOM 对 象 
© load('book xml' ) 加 载 一 个 指定 的 XML 文件 到 内 存 
getElementsByTagName("book" ) 获取 一 个 指定 节点 的 节点 对 象 
item(0) 获取 指定 索引 值 的 节点 对 象 
局 nodeValue 节点 值 
EE 
下 面 创建 一 个 案例 ， 演 示 遍 历 XML 文档 中 的 数据 。 首 先 编写 XML 文件 ， 其 代码 如 下 所 示 : 
六 <?xml] version="1.0" encoding="GB2312"?> 
如 <books> 
完 <book id="1"> 
全 <author sex=" 男 "> 王志刚 </author> 
汪 <publisher> 人 民 邮 电 出 版 社 </publisher> 
阐 <title>JAVA</title> 
册 </book> 
<book> 


<author sex=" 男 "> 如 小 强 </author> 
<publisher> 海 燕 出 版 社 </publisher> 
<tit1le>JSP</tit1le> 

</book> 

</books> 


将 上 述 文件 保存 ， 文 件 名 为 book.xml 文件 ， 将 该 文件 保存 在 C:\Web\apache\htdocsxml 目录 下 ， 
保存 时 字符 集 为 ANSI (默认 状态 )。 然 后 编写 PHP 脚本 程序 ， 其 代码 如 下 所 示 : 


案例 15-2 

<?php 
$doc = new DOMDocument (); 
$doc->load( 'book.xml’' ); 
$books = $doc->getElementsByTagName( "book" ); 
foreach( $books as $book ) 
. 
$authors = $book->getElementsByTagName( "author™ ); 
Sauthor = $authors->item(0) ->nodeValue; 
S$publishers = $book->getElementsByTagName( "publisher™ ); 
Spublisher = $publishers->item(0)->nodeValue; 
$titles = $book->getElementsByTagName( "title™ ); 
$title = $titles->item(0)—->nodeValue; 
echo "$title - $author - $publisher \n<br>"; 
} 


有 过 
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将 上 述 代码 保存 ， 文 件 名 为 Example php。 将 该 文件 和 book.xml 文件 保存 在 同一 个 目录 下 。 打 
开 正 浏 览 器 ,在 地 址 栏 中 输入 http://localhost:8080/ 
xmlExample.php， 单 击 【 转 到 】 按 钮 ， 会 显示 如 图 15-3 
所 示 的 窗口 。 

在 本 案例 中 , 首 先 使 用 语句 “$doc=new DOMDocument 
0” 创 建 了 一 个 DOM 对 象 , 语句 “$doc->load( ee 
表示 加 载 一 个 指定 的 XML 文件 到 内 存 中 ， 并 形成 一 个 
点 树 。 节点 树 形成 之 后 ， 就 可 以 对 节点 进行 操作 了 ， 合用 

语句 “$doc->getElementsByTagName("book")” 创 建 了 一 个 节点 集 对 象 books， 这 时 该 对 象 就 指向 了 

节点 树 中 节点 名 称 book 的 所 有 节点 及 其 子 节点 的 集合 。 该 对 象 可 以 作为 一 个 数组 处 理 ， 在 foreach 
循环 语句 中 ， 将 节点 book 及 其 子 节点 一 一 输出 ， 其 中 语句 “$authors->item(0)-> nodeValue ”表示 获 
取 authors 节点 的 文本 数值 。 

2. 修改 节点 

如 果 对 XML 文档 中 的 节点 或 者 属性 进行 修改 ， 需 要 使 用 到 如 表 15-7 所 示 的 函数 。 

表 15-7 DOM Functions 函 数 


15-3 遍历 XML 


函数 说 阴 
domxpath($dom) 获取 一 个 搜索 对 象 
query("/books/book/title") 获取 指定 路 径 的 节点 对 象 
createElement("title") 创建 一 个 名 称 为 title 的 节点 
appendChild("name") 追加 一 个 内 容 为 name 的 子 节点 
replaceChild($newTitle. $firstTitle) 用 newTitle 节点 替换 firstTitle 节点 
setAttribute("sex", "aaa”) 设置 属性 sex 的 值 为 aaa 
save("newfile.xml") 将 节点 树 保存 为 newfile.xml 文件 
getAttribute("title") 获取 属性 title 的 值 


下 面 创 建 一 个 案例 , 演示 修改 XML 文件 中 节点 和 节点 属性 。 该 案例 采用 的 XML 文件 为 book.xml 
文件 。 现 在 创建 PHP 脚本 程序 ， 其 代码 如 下 所 示 : 


案例 15-3 
<?php 
$dom = new DOMDocument ("1.0"); 
$dom->load ("book.xml"); 
$xp = new domxpath ($dom); 
// 从 DoM 中 修改 节点 数据 
// 修 改 第 一 个 title 的 文件 
// 新 创建 一 个 节点 ， 然 后 替换 旧 的 节点 
$firstTitle = 5Sxp->query("/books/book/title")->item(0) 
SnewTitle = $dom->createElement ("title"); 
SnewTitle->appendchild (new DOMText ("This's the new title text!!!")); 
S$SftirstTitle->parentNode->replaceCchild(SnewTitle，S$SfirstTitle) 7 
// 修 改 属性 
$firstTitlel = $xp->query("/books/book/author") ->item(0) 7 
$firstTitlel->setAttribute ("sex", "aaa"); 
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$dom->save ("newfile.xml")7 
echo $firstTitlel->getAttribute("title"); 
echo "<hr/><a href=\"newfile.xml\"> 查 看 newfile-xml</a>"7 


将 上 述 代码 保存 , 文件 名 为 Example2.php。 打开 正 浏览 器 , 在 地 址 栏 中 输入 http:/localhost:8080/ 
XmlIExample2.php， 单 击 【 转 到 】 按 钮 ， 会 显示 如 图 15-4 所 示 的 窗口 。 
单 击 【查看 newfile xml】 超 级 链接 ， 会 显示 如 图 15-5 所 示 的 窗口 。 


ior sex="'aaa"> 王 者 刚 </author> 


忆 httpz//localhostz8080/zml/Eraple2.php--- 居 | 右 | 区 <punlsher> 人 民 寺 让 出厂 社 </publsher> 


ETTRCTTTIETTIEZTIETTITIETTI <title>Thls’s the new title text lH1</He> 
< 


由 四 圈 weiocaustaapma/awneeoy ”于 加 ee 
<author sex=' 男 '> 吕 小 更 </author> 
<publsher> 海 下 出版 社 </publznor> 
pr <title>ISp</tte> 


图 15-4 修改 XML 节点 图 15-5 修改 后 显示 


在 本 案例 中 创建 了 DOM 对 象 dom, 并 将 book.xml 文件 加 载 到 内 存 中 形成 节点 树 。 使 用 语句 “new 
domxpath($dom)” 创 建 了 一 个 查询 对 象 p， 该 对 象 可 以 获取 节点 树 中 任 一 位 置 的 节点 。 在 下 面 使 用 
方法 query0 获 取 “/books/book/title” 路 径 下 的 节点 对 象 ， 语 句 createElement("title") 表 示 创 建 一 个 新 
的 节点 ， 节 点 名 称 为 tile。 使 用 appendChild0 方 法 为 新 的 节点 添加 一 个 文本 节点 ， 文 本 节点 是 标记 
之 间 的 内 容 。 在 下 面 使 用 replaceChild0 方 法 用 新 建 的 节点 newTitle 蔡 换 旧 的 节点 firstTitle。 这 段 代 
码 表示 修改 节点 的 内 容 。 

如 果 要 修改 某 个 节点 的 属性 值 ， 应 使 用 语句 “$xp->query("/books/book/author")->item(0)” 获 取 
该 节点 对 象 。 获 取 完 成 后 ， 可 以 直接 对 该 对 象 进行 操作 ， 如 使 用 语句 “setAttribute("sex", "aaa")” 设 
置 属性 sex 的 值 为 aaa。 语句 “$dom->save("newfile.xml")” 表 示 将 内 存 中 的 节点 树 保存 为 newfile.xml 
文件 。 

3. 添加 节点 

在 一 个 现 有 的 XML 文档 中 ， 利 用 PHP 程序 添加 新 的 节点 或 新 的 属性 。 需 要 使 用 到 如 表 15-8 
所 示 的 函数 。 

表 15-8 DOM Functions 函 数 
函数 说 明 
createAttribute("id") 创建 一 个 名 称 为 id 的 属性 
createTextNode("3") 创建 一 个 内 容 为 3 的 文本 节点 


下 面 创 建 一 个 案例 ， 该 案例 中 使 用 的 XML 文件 为 book.xml， 演 示 使 用 PHP 程序 在 XML 文档 
中 添加 一 个 节点 或 属性 。 其 代码 如 下 所 示 : 
案例 15-4 


<?php 
$doc = new DOMDocument ("1.0"); 


$doc->load ("book.xml"); 

// 创 建 节点 book 

Snode =$doc->createElement ("book"); 

// 新 建 节点 book 的 属性 id 
$id=$doc->createAttribute ("id"); 
$idText=$doc->createTextNode ("3"); 
$id->appendChild ($idText); 
$node->appendchild ($id); 

// 创 建 book 节点 的 子 节点 author 
$author=$doc->createElement ("author"); 
$author=$node->appendChild ($author); 
S$SauthorText=$doc->createTextNode ("df"); 
$author->appendChild ($authorText); 

// 创 建 book 节点 的 子 节点 title 
$title=$doc->createElement ("title"); 
$title=$node->appendChild ($title); 
$titleText=$doc->createTextNode ("C++"); 
$title->appendChild ($titleText); 

// 将 book 节点 添加 到 XML 文件 中 
$doc->getElementsByTagName ('book')->item(0)->appendChild ($node); 
$doc->save ("nn.xml"); 

echo "<hr/><a href=\"nn.xml\"> 查 看 nn.xml</a>"; 
在 


将 上 述 代 码 保存 ， 文 件 名 为 Example3.php。 打 开 正 浏览 器 ， 在 地 址 栏 中 输入 http:/ 
localhost:8080/xml/Example3.php， 单 击 【 转 到 】 按 钮 ， 会 显示 如 图 15-6 所 示 的 窗口 。 
单 击 【 查 看 nn.xml】 超 级 链接 ， 会 显示 如 图 15-7 所 示 的 窗口 。 
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< verscn="1.0" encoding="GB2312" 1> 
<boai 


=' 昂 "> 王志刚 </author> 
是 电 出 版 社 </Dublshar> 


~ sbook d="3'> 
<author>df</author> 
<title>C++</titlo> 
</book> 
/bcok> 
; ~ <boak> 
ee BW RW IRD PHO <author sex=' 昂 > 吕 小 强 </author> 
ET <publsher> 笨 划 出 版 社 </publisner> 
<title>JSP</title> 
cftcoky 
</oooks> 


15-6 添加 XML 节点 图 15-7 XML 文件 显示 


在 本 案例 中 ， 使 用 语句 “$doc->createElement("book")” 创 建 一 个 节点 名 称 为 “book” 的 节点 对 
象 node,“ $id=$doc->createAttribute("id")” 语 句 表 示 创 建 属性 对 象 d， 并 使 用 “ $idText=$doc-> 
createTextNode("3")” 语 句 创 建 了 一 个 内 容 为 3 的 文本 节点 对 象 idText， 在 下 面 把 节点 对 象 idText 追 
加 到 节点 对 象 ia， 再 把 属性 对 象 id 追加 到 节点 node 对 象 上 。 如 果 要 创建 node 的 子 节点 ， 其 过 程 与 
创建 node 的 属性 基本 一 样 ， 只 不 过 需要 使 用 方法 createElementO 而 不 是 createAttribute0。 最 后 ， 使 
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用 appendChild0 方 法 把 node 节点 对 象 添加 到 XML 文档 中 。 

4. 删除 节点 

如 果 要 删除 XML 文档 中 的 节点 或 属性 ，PHP 脚本 程序 需要 使 用 removeChild0 和 
TemoveAttribute() 方 法 ， 这 些 方法 分 别 表示 移 除 指定 的 节点 和 属性 。 

下 面 创 建 一 个 案例 ， 演 示 删 除 节点 ， 该 案例 使 用 的 XML 文件 为 book xml 文件 。PHP 脚本 程序 
如 下 所 示 : 


案例 15-4 

<?php 
$dom = new DOMDocument ("1.0"); 
$dom->load ("book.xml1"); 
$xp = new domxpath ($dom); 
$titles = $xp->query("/books/book[@id = \"1\"]"); 
$dom->documentElement->removeChild($titles->item(0)); 
$dom->save ("ndd.xml"); 

echo "<hr/><a href=\"ndd.xml\"> 查 看 ndd.xml</a>"; 
KE 


将 上 述 代 码 保存 ， 文 件 名 为 Example4.php。 打 开 正 浏览 器 ， 在 地 址 栏 中 输入 http:/ 
localhost:8080/xml/Example4.php， 单 击 【 转 到 】 按 钮 ， 会 显示 如 图 15-8 所 示 的 窗口 。 
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ET 
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-<be 
<author sex=' 男 '> 吕 小 强 </author> 
<publsher> 海 复出 版 社 </publsner> 


title>]SP<jut 
查看 ndd. xml eapcute> 


</books> 


图 15-8 删除 节点 图 15-9 XML 显示 


在 本 案例 中 ,创建 了 一 个 查询 对 象 xp， 语 句 “Stitles =$xp->query("/books/book[@id =\"1\"]")” 
表示 获取 指定 路 径 ， 并 且 该 节点 的 属性 id 的 值 为 1 的 节点 对 象 tiles 。 然 后 直接 使 用 
“$dom->documentElement->removeChild($titles->item(0))” 移 除 就 可 以 了 。 

使 用 DOM 接口 对 象 操作 XML 文件 ， 还 有 其 他 操作 ， 如 创建 一 个 新 的 XML 文件 等 。 这 里 就 不 
再 介绍 了 ， 有 兴趣 的 读者 可 以 查询 PHP 中 文 手册 学 习 。 


15.2.3 ”使 用 SimpleXML 处 理 XML 


XML 在 数据 管理 和 应 用 程序 之 间 数据 交换 的 操作 性 方面 跃进 了 一 大 步 ， 这 是 所 有 人 的 共识 。 
XML 文件 的 解析 有 多 种 不 同 的 方法 ， 如 DOM、SAX、XSLT 等 ， 对 于 这 些 解析 方法 ， 都 需要 花费 
很 多 精力 和 大 量 的 时 间 来 学 习 XML。 这 对 于 想 利用 XML 解决 问题 ， 但 又 不 懂 XML 的 用 户 设置 了 
一 个 很 大 的 难题 。 在 PHP 5.0 中 ， 提 供 一 个 SimpleXML 函数 库 ， 可 以 解决 这 类 问题 ， 不 需要 太 了 解 
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XML 就 可 以 利用 PHP 程序 解析 XML 文档 。 默 认 情 况 下 该 函数 库 是 可 以 使 用 的 。 
下 面 创 建 一 个 XML 文件 ， 作 为 本 案例 操作 的 对 象 。 打 开 记 事 本 ， 输 入 下 列 内 容 : 


<?xml version="1.0" encoding="GB2312"?> 
<books> 

<book id="1"> 

<author sex=" 男 "> 王志刚 </author> 
<publisher> 人 民 邮 电 出 版 社 </publisher> 
<title>JAVA</title> 

</book> 

<book> 

<author sex=" 男 "> 如 小 强 </author> 
<publisher> 海 燕 出 版 社 </publisher> 
<title>JSP</title> 

</book> 

</books> 


将 上 述 文件 保存 ， 文 件 名 为 book.xml。 
在 SimpleXML 函数 库 中 , 存在 大 量 的 函数 可 以 对 XML 文档 进行 操作 ， 其 中 比较 重要 的 函数 为 
Simplexml load file0。 该 函数 的 语法 格式 如 下 所 示 : 


object Simplexml load file ( string $filename [, string $class_name [, int $options 
[, string $ns [, bool $is prefix]]]] ) 


此 函数 将 flename 指定 的 XML 文件 加 载 到 内 存 。 如 果 加 载 文件 时 遇 到 问题 ， 则 返回 False。 其 
使 用 示例 如 下 所 示 : 
<?php 
$xml=Simplexm] load file("book.xml"); 


Var dump ($xml); 
混交 


将 上 述 代 码 保存 ， 其 保存 位 置 与 book.xml 在 同一 个 目录 下 。 该 段 代 码 执行 的 结果 是 返回 一 个 数 
组 对 象 ， 如 下 所 示 : 


object (SimpleXMLElement)#1 (1) { ["book"]=> array(2) { [0]=> 

object (SimpleXMLElement)#2 (4) { ["@attributes"]=> array(1) { ["id"]=> string(1) 
"I" } ["author"]=> string (9) "王志刚 ”["publisher"]=> string(21) "人 民 邮 电 出 版 社 " 
["title"]=> string(4) "JAVA" } [1]=> object (SimplexMLElement)#3 (3) { ["author"]=> 
string (9) " 吕 小 强 " ["publisher"]=> string (15) "海燕 出 版 社 ” ["title"]=> string (3) 
SA 


当 在 PHP 程序 中 加 载 XML 文档 后 ,就 可 以 使 用 SimpleXML 函数 库 中 的 方法 操作 XML 文档 了 ， 
常用 的 方法 有 4 个， 其 详细 信息 如 表 15-9 所 示 。 

1. attributes() 

XML 属性 提供 了 XML 标记 的 额外 信息 ， 如 果 要 获取 XML 文档 中 某 节 点 的 属性 信息 ， 可 以 利 
用 该 方法 。 该 方法 的 使 用 示例 如 下 所 示 : 
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表 15-9 SimpleXML 常 用 函数 


功 能 
获取 某 节 点 的 属性 信息 
返回 整个 XML 文件 
获取 指定 的 子 节点 
根据 值 有 选择 性 地 获取 节点 


名 称 语法 格式 


attibutes0 
asXMLO mixed asXML ( [string $filename] ) 

children() | SimpleXMLElement children ( [string $ns [. bool $is prefix]] ) 
xpath() 


array xpath ( string $path ) 


<?php 
$xml=Simplexm]l load file("book.xml"); 
foreach ($xml->book as $book){ 
echo $book->author."is".$book->author->attributes().".<br>"; 


} 

2 

在 上 述 代码 中 ， 首 先 加 载 book xml 文件 创建 了 xml 对 象 ， 在 foreach 循环 中 ， 在 名 称 为 book 
的 标记 集合 中 获取 book 标记 的 详细 信息 ， 然 后 输出 标记 book 的 子 标记 author 及 其 属性 。 其 输出 结 
果 为 “王志刚 is 男 . 吕 小 强 is 男 .”。 这 里 需要 注意 的 是 采用 了 面向 对 象 的 写法 。 如 果 一 个 节点 含有 多 
个 属性 ， 可 以 使 用 for 循环 输出 各 个 不 同 的 属性 。 

2. asXML() 

此 方法 基于 SimpleXML 对 象 返回 一 个 格式 良好 的 XML 1.0 字符 串 , 即 返 回 整个 XML 文件 , 但 
删除 XML 文件 中 的 换行 符 。 该 方法 的 使 用 示例 如 下 所 示 : 

<?php 

$xml=Simplexm] load file("book.xml"); 


echo htmlspecialchars ($xm]l->asXML()); 
?> 


3. children() 
该 函数 主要 用 来 获取 指定 的 子 节点 。 其 常用 的 示例 如 下 所 示 : 
<?php 
$xml=Simplexml load file("book.xm]l"); 
foreach ($xml->book[0] ->children() as $character){ 
echo "$character</br>"; 


j 

?> 

上 述 代码 的 执行 结果 返回 该 节点 下 所 有 子 节点 的 文本 信息 ， 如 “王志刚 人 民 邮 电 出 版 社 
JAVA”。 

4. xpath() 

xpath 是 一 个 W3C 标准 ， 它 提供 了 标记 XML 节点 的 一 种 基于 路 径 的 语法 ， 相 当 直观 。 例 如 ， 
引用 books .xml 文档 ， 可 以 使 用 表达 式 /books/book/author 获取 所 有 author 节点 。xpath 还 提供 了 一 组 
函数 ， 可 以 根据 值 有 选择 性 地 获取 节点 。 该 方法 的 使 用 示例 如 下 所 示 : 


<?php 


Sxml1=Simplexml load file("book.xml"); 
$book=$xml->xpath ("/books/book/author"); 
foreach ($book as $book){ 

echo "$book<br/>"; 
} 


ya 


在 上 述 代码 中 , 可 以 获取 所 有 author 标记 中 的 文本 内 容 。 如果 依 据 标记 的 属性 值 选择 一 个 节点 ， 
可 以 使 用 下 面 的 方式 : 


//$book=$xml->xpath ("/books/book/author [sex=\" 男 \"]") 7 
$book=$xml->xpath ("/books/book/author[sex=' 男 ']"); 


在 SimpleXML 函数 库 中 ， 还 有 其 他 的 函数 和 方法 ， 其 使 用 方法 和 上 面 的 方法 比较 相似 ， 这 里 
就 不 再 介绍 了 。 


\ 
@ | 需要 注意 的 是 ， 要 使 用 SimpleXML， 需 要 禁用 PHP 指令 zend.zel_ compatibility mode。 


示 


15.3 ”客户 端 处 理 XML 


使 用 PHP 脚本 程序 处 理 XML 文件 ， 是 在 网 站 的 服务 器 端 进 行 的 。 如 果 所 有 的 处 理 都 在 服务 器 
端 进 行 ， 就 会 加 大 服务 器 端的 工作 量 ， 减 慢 服 务 器 的 执行 速度 。 为 了 减轻 服务 器 的 工作 量 ， 可 以 把 
一 些 不 太 重 要 的 信息 放 在 客户 端 进行 处 理 ， 如 提交 的 信息 格式 等 。 同 样 还 可 以 在 客户 端 处 理 XML 
文件 ， 这 里 通常 采用 的 脚本 语言 是 JavaScript。 

JavaScript 是 一 种 基于 对 象 和 事件 驱动 并 具有 安全 性 能 的 脚本 语言 ， 主 要 用 在 Internet 上 。 使 用 
它 的 目的 是 与 HTML 超 文本 标识 语言 、Java 脚本 语言 一 起 实现 在 一 个 网 页 中 链接 多 个 对 象 , 与 网 络 
客户 交互 作用 ， 从 而 开发 客户 端的 应 用 程序 。 它 是 通过 嵌入 或 调 入 在 标准 的 HTML 语言 中 实现 的 。 
JavaScript 既 可 以 用 到 客户 端 又 可 以 用 到 服务 器 端 。 服 务 器 端 和 客户 端 JavaScript 共享 相同 的 核心 
语言 。 

使 用 JavaScript 解析 XML 文档 ， 可 以 进行 遍历 、 添 加 、 删 除 、 修 改 等 操作 。JavaScript 脚本 的 
执行 通常 是 嵌入 到 HTML 文件 中 的 。 

下 面 创建 一 个 案例 , 演示 使 用 JavaScript 获取 节点 和 相应 的 值 。 该 案例 采用 的 XML 文件 是 上 一 
节 的 book.xml。 打 开 记事 本 ， 输 入 下 列 内 容 : 


案例 15-5 
<script language="JavaSscript"> 
Var xmlDocument = new ActiveXObject ("Microsoft.XMLDOM"); 
xmlDocument .load ("book.xm]l"); 
Var obj=xmlDocument .documentElement .childNodes; 
for (Var i=0;i<obj.length;i++){ 
document .write (obj .item(Ii) .childNodes.item(0) .nodeName+": ") 7 
document .write (obj.item(i) .childNodes.item(0) .text+" ad 
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document .write (obj .item(i) .childNodes.item(1) .nodeName+": "); 
document .write (obj.item(Ii) .childNodes.item(1) .text+" Bd 
document .write (obj.item(i) .childNodes.item(2) .nodeName+": "); 
document .write (obj .item(i) .childNodes.item(2) .text+"<br>"); 
} 
</script> 


将 上 述 代码 保存 ,文件 名 为 Jahtml。 该 文件 要 和 book.xml 文件 保存 在 同一 个 目录 下 。 双 击 Jahtml 
文件 ， 执 行 结果 如 图 15-10 所 示 。 


名 c:AWeb\apacheVhtdoczVzalVJa htal — Eicrosoft Internet Erplorer 


ET TT 人 
地 福 四 | 问 coWebhapsehe hidoes\nl\Je hinl Bl > be) 


author: 王志刚 publisher: 人 民 邮 电 出 版 社 title: JAVA 
author: 昌 小 强 publisher: 海燕 出 版 社 title: JSP 


图 15-10 显示 XML 节点 


在 本 案例 中 ， 语 句 “new ActiveXObject("Microsoft.XMLDOM")” 表 示 创 建 一 个 DOM 对 象 ， 然 
后 使 用 函数 load0 加 载 指定 的 XML 文件 到 内 存 中 进行 操作 ,，“ xmlDocument.document- 
Element.childNodes ”语句 表示 创建 根 节点 下 的 所 有 节点 的 集合 。 其 中 childNodes 表示 子 节 点 的 集合 ， 
item0 方 法 表示 依据 节点 具有 的 子 节点 的 索引 值 返回 该 子 节点 , 属性 nodeName 表示 节点 的 名 称 ， 即 
标记 的 名 称 ， 属 性 text 表示 标记 的 内 容 。 在 上 述 代 码 的 注释 部 分 中 ，xml 表示 返回 整个 XML 文档 ， 
firstChild 表示 获取 该 XML 文档 的 第 一 个 节点 对 象 ， 同 样 有 lastChild 属性 。 


天 室 设 计 


三 Y 学 习 目 标 |ouecove 


随 着 网 络 时 代 的 到 来 ， 越 来 越 多 的 人 开始 上 网 ， 而 大 多 数 上 网 者 所 学 会 的 第 一 件 事 就 是 聊天 。 
聊天 室 的 应 用 在 网 络 发 展 过 程 中 起 到 不 可 估量 的 作用 。 随 着 PHP 语言 的 流行 ， 各 种 样式 的 用 PHP 
语言 编写 的 聊天 室 出 现在 Intemet 上 。 其 实用 PHP 编写 聊天 室 非常 简单 ， 它 类 似 于 前 面 的 交互 式 系 
统 ， 也 可 以 分 为 基于 文本 文件 的 聊天 室 和 基于 数据 库 的 聊天 室 ， 本 章 将 介绍 前 者 。 


人 和 内 容 摘要 1cwct 


了 解 聊天 室 系统 的 实现 功能 

掌握 基于 文本 文件 的 用 户 注册 
掌握 基于 文本 文件 的 用 户 登 录 
理解 聊天 室 主 界面 的 设计 理念 
理解 基于 文本 文件 的 聊天 内 容 显示 
掌握 在 线 用 户 列表 的 显示 

掌握 输入 聊天 内 容 界面 的 设计 及 处 理 
掌握 聊天 室 注销 的 实现 原理 


16.1 系统 概述 


聊天 室 其 实 就 是 多 人 共同 使 用 的 PHP 程序 ， 也 是 Web 中 最 常见 的 服务 项 目 之 一 。 虽 然 各 个 网 
站 中 的 聊天 室 在 界面 上 、 功 能 上 有 着 各 种 变化 ,但 是 其 实现 的 原理 都 是 非常 类 似 的 。 在 Web 聊天 室 
中 ， 使 用 的 是 无 连接 的 HTTP 协议 ， 为 了 能 够 随时 更 新 用 户 输入 的 最 新 聊天 信息 ， 使 用 了 一 种 叫做 
“Client Pull” 的 技术 。 

Netscape 在 3.0 版 本 浏览 器 之 后 使 用 了 新 的 技术 ， 而 Internet Explorer 也 实现 了 这 些 由 Netscape 
公司 开发 出 来 的 技术 。Netscape 公司 将 它 分 成 Server Push 及 Client Pull 两 种 技术 。Server Push 由 
Web 服务 器 将 资料 以 多 重 MIME 编码 发 送 给 客户 端 ， 目 前 使 用 这 种 方式 的 网 站 不 多 ; 而 Client Pull 
则 利用 HTML 的 <meta> 标 签 ， 并 利用 http-equiv=“Refresh” 的 属性 ,通知 服务 器 表示 网 页 要 重新 载 
入 ， 对 于 载 入 时 间 ， 则 利用 content 属性 来 实现 。<meta> 标 签 通 常 都 放 在 <head>...</head> 之 间 ， 以 
便 让 浏览 器 可 以 尽早 准备 更 新 用 户 端 的 网 页 。 
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除 此 之 外 ， 由 于 需要 定期 更 新 所 有 网 友 的 聊天 信息 ， 为 了 避免 写 了 一 半 因 为 刷新 而 被 清除 掉 尚 
未 写 好 的 字符 串 ， 因 此 将 聊天 室 的 页 面 以 框架 网 页 技术 来 实现 是 非常 必要 的 。 

对 于 不 提供 数据 库 支 持 的 网 站 ， 可 以 通过 灵活 运用 对 文本 文件 的 存储 来 修改 实现 聊天 室 的 功 
能 。 文 本 型 聊天 室 的 实现 非常 简单 ， 只 要 每 个 用 户 提交 发 言 ， 系 统 就 用 专门 的 程序 把 发 言 存 到 一 个 
文本 文件 中 ， 然 后 通过 另 一 个 页 面 来 按 设置 的 时 间 读 取 它 ， 再 显示 到 客户 机 的 浏览 器 上 就 行 了 。 

在 本 章 的 案例 中 ， 开 发 一 个 名 叫 “ 爱 岩 聊 天 室 ” 的 聊天 室 程序 。 在 功能 方面 ， 考 虑 到 代码 理解 
的 难 易 程度 ， 这 里 并 没有 选择 介绍 功能 非常 复杂 的 聊天 室 ， 但 是 本 案例 基本 上 实现 了 聊天 室 的 基本 
功能 。 它 主要 实现 以 下 功能 : 

e。 用 户 可 以 注册 新 的 账号 。 

用 户 可 以 登录 聊天 室 。 

用 户 可 以 退出 聊天 室 。 

用 户 在 登录 、 退 出 聊天 室 时 都 登记 有 相应 的 记录 。 

登录 用 户 可 以 在 聊天 室 发 言 ， 游 客 只 能 浏览 其 他 人 的 聊天 记录 ， 而 不 能 进行 发 言 。 
支持 发 言 时 选择 字体 的 颜色 以 及 想 要 聊天 的 对 象 。 


16.2 用户 注 册页 面 


用 户 注册 时 ， 当 用 户 输 入 相应 的 信息 后 ， 单 击 【 我 要 注册 】 按 钮 ， 确 认 进 行 注册 。 这 时 ， 如 果 
要 求 必须 输入 的 信息 没有 输入 或 两 次 输入 的 密码 不 一 致 ， 系 统 都 将 给 出 报错 提示 信息 。 用 户 注册 成 
功 后 系统 将 显示 相应 的 信息 ， 如 果 当 前 提交 的 信息 中 用 到 的 用 户 名 称 已 经 存在 ， 那 么 注册 失败 ， 并 
将 返回 报错 的 提示 信息 。 

registerphp 页 面 用 于 收集 用 户 的 信息 ， 这 里 只 需 用 户 输入 注册 的 用 户 名 和 密码 即 可 。 另 外 ， 通 
过 读 取 文 本 文件 online.txt 中 的 数据 来 判断 聊天 室 当 前 在 线 的 人 数 ， 并 将 其 显示 出 来 。 它 的 具体 代码 
如 下 所 示 : 


<meta http-equiv="Content-Type" content="text/html; charset=gb2312" /> 
<title> 爱 岩 聊天 室 用 户 注册 </title> 
<style type="text/css"> 
<h= 
.STYLE1 { 
Color: #FF0000; 
font-weight: bold; 


一 -> 
</style> 
<script language=javascript> 
function check() 
{ 
if (document .regform.name.value=="'') 
{ 
alert (' 用 户 名 不 能 为 空 !' ) ; 


document .regform.name.focus(); 


return false; 

} 

if(document .regform.pwd.value=="") 
alert ('" 密 码 不 能 为 空 !' ) ; 
document .regform.pwd.focus () 7 
return false; 

} 

if (document .regform.pwd.value!=document .regform.repwd.value) 

和 
alert (' 两 次 输入 的 密码 不 一 致 !') ; 
document .regform.repwd.focus (); 
return false; 


3 
</script> 
<table width="778" style="border:lpx solid;border-color:#000000"> 
<tr> 
<td style="border:lpx solid;border-color:#000000"> 
<img alt=" 爱 岩 聊天 室 " align="middle" src="img/top.jpg"> 
</td> 
</tr> 
<tr> 
由 
<td> 
<a href="http://www.itzcn.com">IT 在 中 国 </a> 
<a href="http://www.itying.net"> 技 术 学 堂 </a> 
</td> 
类 让 > 
<tr> 
<td style="border:lpx solid;border-color:#000000"> 
<form action="regok.php" method="post" name="regform"> 
<div align="center" class="STYLE1"> 爱 岩 聊天 室 用 户 注册 </div> 
<hr noshade style="border:lpx solid;border-color:#000000"> 
<p align="center"> 请 输入 你 希望 使 用 的 用 户 名 : 
<input name="name" type="text" id="name"> 
</p> 
<p align="center"> 请 输入 你 希望 使 用 的 密码 : 
<input name="pwd" type="password" id="pwd"> 
</p> 
<p align="center"> 请 再 次 输入 你 希望 使 用 的 密码 : 
<input name="repwd" type="password" id="repwd"> 
</p> 
<div align="center"> 
<p> 
<input type="reset"” value=" 重 填 " > 
<input type="submit"™" value=" 我 要 注册 " onClick="check()" > 
</p> 
<p><strong> 
<a href="chat .php"> 我 要 以 [游客 ] 身份 登录 </a> 


</strong> </p> 


互 滥 
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regok.php 页 面 用 于 将 由 registerphp 页 面 传递 过 来 的 用 户 信息 添加 到 文本 文件 usertxt 中 。 在 插 
入 记录 之 前 要 进行 检查 ， 如 果 存 在 相同 的 用 户 名 ， 则 注册 失败 。 它 的 具体 代码 如 下 所 示 : 


if(Sname == Soname[0]) 
echo " 爱 岩 聊天 室 提示 : <hr><p>"; 
echo "对 不 起 ， 该 用 户 名 已 被 注册 ， 你 不 能 重复 注册 ! "; 


echo "<br><a href='register.php'> 请 重新 进行 注册 </a>"; : 
echo "<br><a href='chat .php'> 或 以 [游客 ] 身份 登录 </a>"; 
Srepeat = 1; ee 
break; 
| 
} 
// 如 果 当 前 用 户 名 未 曾 注册 
// 写 入 注册 用 户 文件 


if($repeat != 1) 
{ 


互 小 


$fp = fopen(Suserfile,"a") 7 
if(!$fp) 


{ 
die ("<br> 不 能 打开 用 户 文件 。") ; 
} 
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fwrite ($fp,"\r\n"); 
fwrite ($fp, $name."&&"); 
fwrite ($fp, $pwd."&&"); 
fclose ($fp); 


echo " 爱 岩 聊 天 室 提示 : <hr><p>"; 

echo "[$name] 用 户 注册 成 功 "; 

echo "<br><a href='index.php'> 请 返回 登录 </a>"; 

echo "<br><a href='chat .php'> 或 以 [游客 ] 身份 登录 </a>"; 


} 
1?> 


把 上 述 文件 registerphp 和 regok.php 存储 在 Apache 目录 下 的 htdocs\chat 子 目 录 下 ， 打 开 正 浏 
览 器 ， 在 地 址 栏 中 输入 http://localhost/chat/register.php， 按 Enter 键 会 显示 如 图 16-1 所 示 的 页 面 。 


下 天 洗 聊 天 室 用 户 注 册 -了 iccozoft Internct Explorer 
文件 2) 编辑 于 查看 W) 收 意 让 ) 工具 0) 略 助 0 
二 址 四 | 虱 http.//1ocnhost/ cai/reeister. phy 


要 兰 枝 天 至 用 户 注 和 贡 


请 输入 你 节 望 使 用 的 用 户 名 ， | 宗 关 尖 


请 输入 你 项 望 使 用 的 密码 。 see 


请 再 次 输入 你 着 望 使 用 的 密码 ， [ee 


Eee 
以 [游客] 身份 桔 : 


当前 在 线 人 数 ， 0 


16-1 用 户 注 册页 面 


387 


在 该 页 面 中 输入 用 户 名 称 “ 宋 岩 岩 ” 密码 及 确认 密码 “bob”， 然 后 单 击 【 我 要 注册 】 按 钮 ， 
如 果 注 册 成 功 会 显示 如 图 16-2 所 示 的 页 面 ; 如 果 用 户 输入 的 用 户 名 已 经 存在 ， 将 注册 失败 ， 并 会 显 
示 如 图 16-3 所 示 的 页 面 。 

各 用 户 注册 执行 结果 证 et Eu] .3 x 加 用 户 注册 执行 续 果 一 看 


EE 工具 中 利 助 0 
地 丰 四 | 阐 My focahosVehar/re 癌 术 加 | 阐 Mey /focalhos Veh/reeok pap 


防 几 央 用 /手机 成 区 对 不 起 ， 该 用 户 名 已 被 注册 ， 人 天 
请 重新 进行 
或 以 | 游 宇 ] 身 记 章 录 或 以 [六 客 ] 身 份 车 录 


ET a at 


图 16-2 用 户 成 功 注册 图 16-3 已 经 存在 同样 的 用 户 名 


16.3 ”用 户 登 录 页 面 


用 户 登 录 页 面 index.php 用 于 已 经 注册 的 用 户 登录 聊天 室 ， 用 户 还 可 以 以 游客 的 身份 进入 聊天 
室 ， 但 不 能 进行 聊天 操作 。 另 外 ， 用 户 还 可 以 单 击 【 我 要 注册 】 超 级 链接 转向 注册 页 面 进行 注册 。 
该 页 面 的 具体 代码 如 下 所 示 : 


<meta http-equiv="Content-Type" content="text/html; charset=gb2312" /> 
<title> 用 户 登录 </title> 
<table width="778" style="border:lpx solid;border-color:#000000"> 
<tr> 
<td style="border:lpx solid;border-color:#000000"> 
<img alt=" 爱 岩 聊 天 室 " align="middle" src="img/top.jpg"> 
</td> 
</tr> 
<tr> 
Am 
<td> 
<a href="http://www.itzcn.com">IT 在 中 国 </a> 
<a href="http://www.itying.net"> 技 术 学 堂 </a> 
</td> 
</tr> 
<tr> 
<td style="border:lpx solid;border-color:#000000"> 
<form action="loginchk.php" method="post"> 
<div align="center"><strong> 爱 岩 聊 天 室 用 户 登录 
</strong></div> 
<hr noshade style="border:lpx solid;border-color:#000000"> 
<p align="center"> 请 输入 你 的 用 户 名 : 
<input name="name" type="text" id="name"> 
</p> 
<p align="center"> 请 输入 你 的 密码 : 
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loginchk .php 页 面 用 于 验证 用 户 的 用 户 名 和 密码 是 否 正确 。 如 果 用 户 名 和 密码 与 文本 文件 中 的 
某 一 用 户 数据 相对 应 ， 那 么 就 将 该 用 户 的 信息 保存 到 Session 以 备 后 面 操作 的 需要 ， 在 user 目录 下 
更 新 或 创建 用 户 记录 文件 、 更 新 用 户 在 线 文件 online.txt 及 向 msg.txt 文件 写 入 欢迎 信息 ， 最 后 转向 
聊天 室 页 面 文件 hatphp。 和 否则 返回 相应 的 报错 信息 。 该 页 面 的 具体 代码 如 下 所 示 : 


PHP+MySQL 候 包 洲 扩 中 课 


$flag=1; 
continue; 


} 


// 判 断 验证 结果 
if($flag == 2) 
{ 

$_SESSION['name'] = $name; 
$_SESSION['userid'] = $userid; 
$_SESSION['ctime'] = $ctime; 
// 写 入 用 户 登 录 记录 文件 
S$ufile = "user/$name.txt"; 
$fp = fopen($ufile,"a"); 
if(!$fp) 
{ 


已 滥 


die ("<br> 不 能 打开 用 户 文件 。") ; 
} 
$str = "欢迎 用 户 Sname [ $userid ] 于 时 间 [ $ctime ] 进入 爱 岩 聊 天 室 ! "7 
fwrite ($fp, $str); 
fwrite ($fp,"\r\n"); 
fclose ($fp); 


闻 闷 财 州 四 才 


// 写 入 所 有 用 户 的 聊天 记录 文件 
Smsgfile = "msg.txt"; 
$fpmsg = fopen($msgfile,"a"); 
if(!$fpmsg) 
i 
die ("<br> 不 能 打开 用 户 文件 。") ; 
} 
//fseek ($fpms,0,SEEK SET); 
// 设置 输出 格式 
$str = "<b> 聊 天 室 公告 </b> 欢迎 用 户 $name [ $useriqd ] 于 时 间 [ $ctime ] 
进入 爱 岩 聊天 室 ! "; 
Snew_message = "<font color=\"black\">$str<p>\n </font>"; 
Sheader = "<html><head>". 
"<meta http-equiv=\"pragma\" content=\"no-cache\">". 
"<meta name=\"robots\" content=\"noindex\"></head>". 
"<body bgcolor=\"#ffffff\" text=\"#000000\"><p>". 
"<embed src=\"alert.wma\" width=2 height=0 autostart=true 
loop=false>". 
"<noembed><bgsound src=\"alert .wma\" loop=none></noembed>\n"; 
$footer = "</body>"."</html>"; 
fwrite ($fpmsg, $header); 
fwrite ($fpmsg, Snew message); 
fwrite ($fpmsg, $footer); 
fwrite ($fpmsg, "\r\n"); 
fclose ($fpmsg); 
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// 写 入 在 线 用 户 文件 
Sonfile = "online.txt"; 
$fpon = fopen($onfile,"a"); 
if(!$fpon) 
是 
die ("<br> 不 能 打开 在 线 用 户 文件 。") ; 
有 
fwrite ($fpon, $name."&&"); 
fwrite ($fpon,"\r\n"); 
fclose ($fpon); 


// 用 于 转向 chat .php 页 面 

echo "<script language=JavaScript>"; 
echo "location.href='chat.php';"; 
echo "</script>"; 


elseif ($flag == 1) 


{ 
echo 
echo 
echo 
echo 
echo 


} 


7 


"用 户 登 录 失败 ! <hr><p>"; 

"用 户 名 或 密码 不 正确 ! "; 

"<br><a href='index.php'> 请 重新 登录 </a>"; 

"<br><a href='register.php'> 请 先进 行 注册 </a>"; 

"<br><a href='chat .php?1login=yk'> 或 以 [游客 ] 身份 登录 </a>"; 


把 上 述 文件 index.php 和 loginchk.php 存储 在 Apache 目录 下 的 htdocs\chat 子 目录 下 , 打开 正 浏 
览 器 ， 在 地 址 栏 中 输入 http://localhost/chat/index.php， 按 Enter 键 会 显示 如 图 16-4 所 示 的 页 面 。 


I 在 中 国 技术 学 党 


要 加 一 天 室 用 户 橙 孙 


靖 答 入 你 的 月 户 名 ， [FS 攻 


请 给 入 你 的 密码 。%9 


16-4 用户 登录 页 面 


在 该 页 面 中 输入 自己 的 用 户 名 和 密码 ， 然 后 单 击 【 确 定 】 按 钮 进行 登录 。 如 果 用 户 输入 的 用 户 
名 和 密码 不 正确 , 或 者 当前 用 户 还 没有 进行 注册 ， 那 么 会 显示 如 图 16-5 所 示 的 报错 提示 页 面 。 这 里 
使 用 前 面 已 经 注册 的 用 户 名 “ 宋 岩 岩 ” 和 密码 “bob” 进 行 登录 ， 登 录 成 功 以 后 将 转向 chat.php 文件 


所 显示 的 聊天 室 页 面 。 


对 用户 登录 结果 - 了 icrosoft Internet Explorer 
文件 F) 编 加 FE) 查看 收 意 O) 工具 0T) 才 助 00 
说 泪 0) | 罩 http://1ocalhost/chat/loginchk php 


用 户 登录 失败 ! 


用 户 名 或 密码 不 正确 ! 
请 重新 谷 录 


导 行 
或 以 [游客 ] 


图 16-5 错误 提示 页 面 


16.4 ”聊天 室 的 主页 面 


聊天 室 的 主页 面 是 通过 使 用 框架 集 来 呈现 的 ， 之 所 以 这 样 是 为 了 防止 显示 聊天 信息 页 面 的 刷新 
对 其 他 页 面 造成 的 影响 。 它 包含 了 4 个 页 面 。 

1. 顶部 页 面 

该 页 面 通常 情况 下 用 于 显示 网 站 的 Logo 图 标 、 网 站 的 标题 、 网 站 的 导航 菜单 及 其 他 一 些 需 要 
着 重 对 用 户 展示 的 内 容 等 ， 该 页 面 的 实现 文件 为 top.php。 这 里 只 是 为 了 展示 ， 它 的 代码 比较 少 ， 具 
体 如 下 所 示 : 


<table width="778" style="border:lpx solid;border-color:#000000"> 
<tr> 
<td style="border:1lpx solid;border-color:#000000"> 
<img alt=" 爱 岩 聊天 室 " align="middle" src="img/top.jpg"> 
</td> 
</tr> 
<tr> 
<td> 
<a href="http://www.itzcn.com" target=" blank">IT 在 中 国 </a> 
<a href="http://www.itying.net"” target=" blank"> 技 术 学 堂 </a> 
<a href="http://www.itzcn.com/bbs" target=" blank"> 交 流 论坛 </ a> 
<a href="http://www.google.com" target=" blank">Google 搜索 </a> 
<a href="http://www.baidu.com" target=" blank">Baidu 搜索 </a> 
</td> 
</tr> 
</table> 


2. 显示 聊天 内 容 页 面 

该 页 面 通过 读 取 文本 文件 msg.txt 中 的 内 容 来 向 用 户 显示 它们 的 聊天 信息 , 并 且 该 页 面 按照 事先 
设 定 的 时 间 周 期 性 地 进行 刷新 ， 以 显示 用 户 的 最 新 聊天 信息 。 

3. 显示 在 线 用 户 列表 页 面 

该 页 面 通过 读 取 文本 文件 online.txt 中 的 内 容 来 显示 已 经 登录 的 用 户 名 , 该 页 面 同 显示 聊天 内 
页 面 一 样 要 进行 周期 性 的 刷新 ， 以 使 用 户 看 到 最 新 的 在 线 情况 。 

4. 输入 聊天 内 容 页 面 

该 页 面 用 于 登录 用 户 输入 聊天 内 容 , 并 将 其 提交 给 后 台 PHP 脚本 处 理 , 最 后 将 聊天 内 容 写 入 聊 


已 站 
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天 内 容 文 本 文件 msg.txt 中 。 

在 上 述 4 个 相关 的 页 面 中 ， 显 示 聊 天 内 容 页 面 、 显 示 在 线 用 户 列表 页 面 和 输入 聊天 内 容 页 面 将 
在 后 面 章节 进行 详细 的 介绍 。 

关于 包含 这 4 个 页 面 的 聊天 室 主页 面 chatphp， 它 本 身 并 不 实现 这 些 页 面 的 功能 ， 而 只 是 把 它 
们 呈现 的 内 容 包 含 进来 而 已。 它 的 具体 代码 如 下 所 示 : 


<meta http-equiv="Content-Type" content="text/html; charset=gb2312" /> 
<title> 爱 岩 聊 天 室 </title> 
<frameset rows="200,*,90" cols="*" frameborder="no" border="0" framespacing="0"> 
<!-- 顶 部 页 面 -> 
<frame src="top.php" name="top" scrolling="No" noresize="noresize" id="top" 
title="top" /> 
<!-- 在 线 用 户 页 面 和 聊天 信息 页 面 --> 
<frameset cols="140,*" frameborder="no" border="0" framespacing="0"> 
<frame src="left.php" name="left" scrolling="no" noresize="noresize" id="]left" 
title="left" /> 
<frame src="msg.php" name="main" id="main" title="main" /> 
</frameset> 
<!-- 输 入 聊天 内 容 的 页 面 --> 
<frame src="send.php" name="foot" scrolling="no" noresize="noresize" id="foot" 
title="foot" /> 
</frameset> 
<noframes> 
<body> 
对 不 起 ， 你 的 浏览 器 不 支持 框架 ! 
</body> 
</noframes> 


把 上 述 文件 chat.php 和 top.php 存储 在 Apache 目录 下 的 htdocs\chat 子 目 录 下 (这 里 假设 上 述 代 
码 用 到 的 leftphp、msg.php 和 send.php 文件 已 经 创建 )， 打开 正 浏览 器 ， 在 地 址 栏 中 输入 
http://localhost/chat/index.php， 按 Enter 键 后 会 显示 用 户 登 录 页 面 ， 输 入 正确 的 用 户 名 和 密码 ， 登 录 
成 功 后 会 显示 类 似 于 如 图 16-6 所 示 的 聊天 室 页 面 。 
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在 线 用 户 a 局 户 末 岩 岩 [ 127. 于 时 介 [ 2007-07-25 05:57 ] 过 出 委 洗 获 天 
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Et 
罕 公 告 。 欢 迎 用 户 龙 建华 【 127.0.0.1 ] 于 时 间 [ 2007- ] 江 入 爱 央 
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告 ， 欢迎 用户 宋 岩 岩 [ 127. 0.0.1 ] 于 时 间 [ 2007-07-25 07:45 ] 


站 :6] 宋 岩 岩 对 所 有 人 说: 大 家 打 用 ，FHF 编 径 学 习 的 如 向 芹 1 


次 迎 用 户 于 品 品 [ 127.0.0.1 ] 于 时 间 [ 2007- 


[2007-07-25 07:45] 王 晶 曲 对 宋 岩 岩 说 : 和 的 好 朋友 ， 你 隶 近 好 四 


文 市 产 色 ;| ”聊天 对 象 。 睫 有 “技术 支 疮 我 可 退出 
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16-6 ”聊天 室 页 面 


16.5 显示 聊天 内 容 页 面 


显示 聊天 内 容 页 面 的 文件 是 msg.php， 它 通过 PHP 脚本 循环 地 读 取 用 户 聊天 信息 文件 msg.txt 
中 的 内 容 ， 并 将 其 显示 出 来 。 它 的 具体 代码 如 下 所 示 : 


<meta http-equiv="refresh" content="30"> 
<meta http-equiv="pragma" content="no-cache"> 
<link href="css.css" rel="stylesheet" type="text/css" /> 
<table width="630" border="0" cellspacing="0" cellpadding="0"> 
<tr> 
<td align="left"> 


< 
// 获得 用 户 聊天 记录 
Smsgfile = "msg.txt"; 
$fpmsg = fopen($msgfile,"r"); 
if(!$fpmsg) 
{ 
die ("<br> 不 能 打开 用 户 文件 。") ; 
} 
Srstr = fgets($fpmsg, 60); 
while(!feof ($fpmsg)) 
{ 
echo $rstr; 
Srstr = fgets($fpmsg, 60); 
} 
fclose ($fpmsg); 
?> 
</td> 
</tr> 
</table> 


下 面 对 上 述 代 码 中 的 两 行 代码 进行 说 明 : 


<meta http-equiv="refresh" content="30"> 
<meta http-equiv="pragma" content="no-cache"> 


第 一 行 用 于 设 定 页 面 的 自动 刷新 时 间 ， 这 里 设 定 的 是 30s， 这 样 该 页 面 就 会 每 隔 30s 重新 被 载 
入 。 第 二 行 则 用 于 不 让 浏览 器 把 该 页 面 存 入 客户 端 缓存 ， 而 是 每 次 载 入 时 从 服务 器 获取 ， 这 样 才能 
保证 当前 显示 的 是 最 新 的 聊天 信息 。 


16.6 显示 在 线 用 户 列表 页 面 


显示 在 线 用 户 列表 页 面 的 文件 是 leftphp, 它 通过 PHP 脚本 循环 读 取 用 户 在 线 信息 文件 online.txt 
中 的 内 容 ， 将 其 显示 出 来 。 它 的 具体 代码 如 下 所 示 : 


<meta http-equiv="refresh" content="60"> 
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16.7 输入 聊天 内 容 页 面 


输入 聊天 内 容 页 面 的 文件 是 send.php， 它 通过 判断 用 户 是 否 是 登录 用 户 来 设置 用 户 名 项 的 值 ， 
以 及 设置 聊天 内 容 的 颜色 和 聊天 的 对 象 ， 然 后 发 送 到 sendok.php 页 面 进行 处 理 。 该 页 面 的 具体 代码 
如 下 所 示 : 


$ SESSION['name'"] ?>" disabled="disabled" > 
<?php 
. 
else 
> 
用 户 昵称 ， <input type="text" name="chatname" size="16" value="<?php echo 
$chatname ?>" disabled="disabled" > 
<?php 
?> 
文本 颜色 : <select name="myfont"> 
<option value="red"> 红 色 </option> 
<option value="black" selected="selected"> 黑 色 </option> 
<option value="blue"> 蓝 色 </option> 
<option value="yellow"> 黄 色 </option> 
<option value="green"> 绿 色 </option> 
<option value="white"> 白 色 </option> 
</select> 
聊天 对 象 : 
<select name="selectuser"> 
<option value=" 所 有 人 "” selected="selected"> 所 有 人 </option> 
<? 
// 获得 在 线 用 户 列表 
S$olfile = "online.txt"; 
S$online array = file($olfile); 
foreach ($online array as $user on) 
{ 
Soneuser = explode("&&", $user on); 
?> 
<option value="<?php echo $oneuser[0] ?>"><?php echo $oneuser[0] ?></option> 
<?php 
} 
?> 
</select> 
&nbsp; &nbsp; &nbsp; gnbsp; <a href="http://www.itying.net" target=" blank"> 技 术 
支持 </a> 
<a href="logout .php” target=" top"> 我 要 退出 </a><br> 
消息 : <input type="text" name="message" size="80">gnbsp; 
<input type="submit" value=" 发 送 "> 
<br> 
</form> </td> 
</tr> 
</table> 


sendok.php 页 面 用 于 接收 send.php 页 面 提交 的 聊天 数据 ， 并 判断 该 数据 是 否 为 登录 用 户 所 提交 
的 数据 。 如 果 是 ，PHP 脚本 程序 将 聊天 数据 写 入 msgtxt 文本 文件 中 ， 并 返回 send.php 输入 聊天 内 
容 页 面 ; 如 果 用 户 未 登录 ， 那 么 将 显示 报错 提示 信息 ， 并 要 求 用 户 进行 注册 或 登录 。 该 页 面 的 具体 
代码 如 下 所 示 : 
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PHP+MySQL 了 骸 促 钉 式 炸 案 


fclose ($fpmsg); 


// 用 于 转向 chat .php 页 面 


echo “<script language=JavaScript>"; a 
echo "location.href='send.php';"; 
echo "</script>"; eo. 
soe 
} . 
} 
2 


登录 用 户 进入 聊天 室 后 可 以 参与 交流 ， 这 里 以 游客 的 身份 进入 聊天 室 ， 并 输入 聊天 信息 ， 然 后 
单 击 【 发 送 】 按 钮 提交 留言 ， 会 显示 如 图 16-7 所 示 的 页 面 。 


台 品 


文件 加 ) 编辑 区) 查理) 收 套 O)。 工具 G) 帮助 00 


章 
聊 
天 
室 
设 
计 


| 技术 学 党 | 运 | Google 搜索 | aaidu 搜 过 I 
Bs 用 户 天 建华 127.0.0.1 ] 于 时 间 [ 2007-07-25 07:53 ] 泪 印 要害 散 天 区 
| 


当前 时 间 ， 聊天 室 公告 ， 用 户 王 晶 曲 【 127. 0.0.1 ] 于 时 间 [ 2007-07-25 07:54 ] 退出 受 岩 聊天 
ea 至 | 


爱 央 黎 天 宝 提示 ， 


你 没有 登录 ， 不 能 进行 聊天 
请 公 录 或 这 行 注册 


图 16-7 游客 不 能 参与 交流 


从 该 图 可 以 看 出 ， 以 “游客 ”身份 参与 交流 是 不 被 允许 的 ， 并 且 所 提交 的 信息 也 不 会 出 现在 显 
示 聊 天 内 容 的 页 面 上 。 


16.8 ”聊天 室 注 销 页 面 


聊天 室 注销 页 面 实现 的 文件 是 logoutphp， 它 用 于 实现 登录 用 户 的 注销 功能 。 首 先 判 断 访问 该 
页 面 的 用 户 是 否 已 经 登录 ， 如 果 是 已 登录 用 户 ， 则 从 登录 用 户 文本 文件 online.txt 中 删除 该 用 户 的 登 
录 信 息 ; 其 次 , 将 用 户 退 出 的 信息 写 入 user 目录 下 并 以 用 户 自己 的 用 户 名 命名 的 登录 文件 中 ; 再 次 ， 
将 用 户 退 出 聊天 室 的 信息 写 入 msg.txt 文本 文件 中 ,以 使 所 有 登录 用 户 知道 ; 最 后 ,破坏 用 户 登录 时 
创建 的 Session。 该 页 面 的 具体 代码 如 下 所 示 : 

<?php 


session start(); 
$chatname = $ SESSION['name']; 


PHP+MySQL 巩 媳 锌 式 由 束 


} 


// 以 写 的 方式 打开 文件 
$fp=@fopen ($onfile, "w"); 
@fputs ($fp, $newfp); 
@fclose ($fp); 


if($flag > 0) 


{ 


// 写 入 用 户 登 录 记录 文件 
S$ufile = "user/$chatname.txt"; 
$fp = fopen($ufile,"a"); 


if(!$fp) 
f 第 
die ("<br> 不 能 打开 用 户 文件 。") ; 加 
T 早 
$str = "用 户 Schatname [ $userid ] 于 时 间 [ $ctime ] 离开 爱 岩 聊天 室 ! "; 聊 
fwrite ($fp, $str); 天 
fwrite(Ssftp; "Nr\n")s? 室 
fclose ($fp); 设 
计 
// 写 入 所 有 用 户 的 聊天 记录 文件 


Smsgfile = "msg.txt"; 

$fpmsg = fopen($msgfile,"a"); 

if(!$fpmsg) 

{ 
die ("<br> 不 能 打开 用 户 文件 。") ; 

} 

fseek ($fpms, 0, SEEK SET); 

// 设置 输出 格式 

$str = "<b> 聊 天 室 公告 :</b> 用户 $Schatname [ $userid ] 于 时 间 [ $ctime ] 退 

出 爱 岩 聊 天 室 ! "; 

$new message = "<font color=\"red\">$str<p>\n </font>"; 

Sheader = "<html><head>". 
"<meta http-equiv=\"pragma\" content=\"no-cache\">". 
"<meta name=\"robots\" content=\"noindex\"></head>". 
"<body bgcolor=\"#ffffff\" text=\"#000000\"><p>"; 

$footer = "</body>"."</html>"; 

fwrite ($fpmsg, $header); 

fwrite ($fpmsg, $new message); 

fwrite ($fpmsg, $footer); 

fwrite ($fpmsg, "\r\n"); 

fclose ($fpmsg); 


echo " 爱 岩 聊天 室 提示 : <hr><p>"; 

echo "你 已 经 成 功 退出 登录 ! "; 

echo "<br><a href='index.php'> 请 重新 登录 </a>"7 

echo "<br><a href='register.php'> 或 进行 注册 </a>"; 

echo "<br><a href='chat.php?1ogin=yk'> 或 以 [游客 ] 身份 登录 </a>"; 
// 破 坏 登 录 时 创建 的 Session 


401 


当 用 户 在 聊天 室 页 面 中 ， 单 击 【 退 出 登录 】 超 级 链接 时 ， 如 果 注 销 成 功 会 显示 如 图 16-8 所 示 的 
页 面 。 这 里 假设 退出 登录 的 用 户 是 宋 岩 岩 ， 那 么 查看 user 目录 下 的 文件 “ 宋 岩 岩 .txt”， 将 会 看 到 类 
似 于 如 下 所 示 的 记录 : 


= pss 


嘻 汕 避 性 悄 沿 19SAW+dHd 


16-8 ”登录 用 户 成 功 注销 
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留言 板 作为 网 站 交互 的 平台 , 已 经 成 为 商务 网 站 必 不 可 少 的 模块 。 在 网 站 平台 上 提出 产品 建议 、 
发 表 评论 都 需要 使 用 到 留言 板 。 常 见 的 留言 板 具 有 添加 留言 、 删 除 留言 、 显 示 留 言 的 功能 。 

本 章 将 详细 地 介绍 实现 留言 板 的 整个 过 程 ， 其 中 重点 介绍 留言 板 数据 表 的 建立 、 发 表 留 言 和 显 
示 留 言 内 容 等 步骤 的 实现 过 程 。 本 章 最 大 的 特色 是 使 用 ODBC 数据 源 实现 对 数据 库 的 操作 。 


区 内 容 摘要 |avstrsct 


> 了 解 留言 板 的 作用 

> 掌握 留言 板 具 有 的 功能 

> 掌握 编写 和 调试 PHP 代码 

> 掌握 在 不 同 的 PHP 页 面 传 值 
熟练 掌握 使 用 ODBC 数据 源 

> 掌握 PHP 页 面 中 的 分 页 显示 

> 掌握 在 PHP 中 使 用 JavaScript 脚本 


17.1 系统 及 数据 库 设 计 


留言 板 在 一 个 网 站 中 是 必 不 可 少 的 部 分 ， 通 过 它 可 以 实现 收集 浏览 者 的 建议 ， 获 取 产 品 的 反馈 
信息 。 在 网 站 运行 时 ， 往 往 都 会 考虑 到 和 浏览 者 之 间 的 互动 关系 。 通 过 留言 板 可 以 实现 这 种 关系 。 
网 站 的 浏览 者 可 以 在 留言 板 上 写 下 自己 对 该 网 站 的 看 法 和 意见 ， 或 者 提出 一 些 问题 等 ， 而 网 站 的 维 
护 人 员 也 可 以 在 留言 板 上 回复 这 些 意 见 和 看 法 ， 就 网 友 提 出 的 问题 进行 回答 ， 由 此 不 难 发 现 留言 板 
的 重要 作用 。 

开发 该 留言 板 所 需要 的 开发 工具 、 运 行 环境 、 数 据 库 ， 分 别 为 纯 文 本 编辑 器 记事 本 、PHP 
5.0+Apache 2.2、Access 2003。 以 记事 本 作为 开发 工具 主要 是 显示 出 PHP 使 用 ODBC 数据 源 连 接 数 
据 库 的 应 用 。 该 留言 板 的 开发 目的 有 两 个 ， 一 是 供 广大 读者 学 习 ， 另 外 是 供 公司 使 用 ， 可 以 在 本 留 
言 板 的 基础 上 加 以 改造 ， 成 为 一 个 商业 性 质 的 网 站 模块 之 一 。 本 留言 板 创建 完成 之 后 ， 要 具备 下 面 
的 功能 : 添加 留言 、 删 除 留言 、 查 看 留言 等 。 

在 Access 数据 库 中 ， 创 建 一 个 表 用 来 存储 浏览 者 所 要 发 表 的 信息 ， 名 称 为 liuyan。 该 表 所 有 的 
字段 如 表 17-1 所 示 。 

数据 库 创建 完成 后 ， 需 要 创建 ODBC 数据 源 。 单 击 【 开 始 】 菜 单 ， 选 择 【 控 制 面板 】| 【管理 
工具 】| ODBC 命令 ， 在 弹出 的 【ODBC 数据 源 管理 器 】 对 话 框 中 选择 【系统 DSN】 选 项 卡 ， 如 图 
17-1 所 示 ， 单 击 【添加 】 按 钮 ， 会 显示 如 图 17-2 所 示 的 对 话 框 。 


Y 


表 17-1 留言 信息 表 


字 段 备 注 
tie id 留言 信息 号 
tie name 留言 人 姓名 
tie title 留言 题目 
tie_content 留言 内 容 


创建 新 数据 源 
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在 图 17-2 中 选择 Access 数据 库 的 驱动 程序 。 选 择 完毕 后 ， 单 击 【 完 成 】 按 钮 会 显示 如 图 17-3 
所 示 的 对 话 框 ， 在 【数据 源 名 】 文 本 框 中 输入 要 创建 的 数据 源 名 称 ， 输 入 完毕 后 ， 单 击 【 选 择 】 按 


钮 ， 会 显示 
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图 17-3 创建 数据 源 图 17-4 选择 数据 库 
至 此 ，ODBC 数据 源 已 经 创建 成 功 。 


17.2 ”留言 主页 面 


在 实现 留言 板 代码 前 ， 需 要 创建 一 个 文件 夹 用 来 保存 要 创建 的 代码 ， 其 位 置 在 
C:\Web\apache\htdocs 文件 夹 下 , 名 称 为 yan, 并 在 yan 文件 夹 下 创建 一 个 CSS 文件 夹 保存 CSS 格式 
文件 。 另 外 在 yan 文件 夹 下 创建 一 个 独立 的 文件 Conn， 该 文件 主要 实现 用 类 来 完成 数据 库 操作 ， 如 
添加 、 删 除 、 修 改 等 。 


打开 记事 本 ， 输 入 下 列 代码 : 


将 上 述 代 码 保存 ， 文 件 名 为 Conn.php。 在 上 述 代码 中 ， 创 建 了 一 个 类 Odbcon， 在 类 中 创建 方 
法 inser0 用 来 执行 相应 的 SQL 语句 ， 如 添加 、 删 除 等 。 
下 面 创建 留言 板 程序 的 主 界面 ， 打 开 记 事 本 ， 输 入 下 列 代码 : 


<H3><A 

href-"cha.php"> 查 看 留言 </A> 

</H3> 

<UL></UL></DIV></DIV></DIV> 

<DIV id=center> 

<DIV class=content> 

<H2> 请 您 留言 </H2> 

<FORM action="success.php" method=post> 

<INPUT type=checkbox value=] name=autoLogin> 署名 发 表 ( 贴 子 可 管理 ,不 必 输 入 下 面 的 姓名 ) 
(AD <BR> 姓 名 : <INPUT name=author> 
<BR> 主 题 : <INPUT maxLength=60 size=45 name=subject> 
<BR> 内 容 : <TEXTAREA name=body rows=15 cols=60></TEXTAREA> 
<BR><INPUT type=submit value= 提 交 name=login> 


rl <INPUT type=reset value= 重 置 name=again> 
汪 </FORM> 
二 <P></P></DIV></DIV> 
Es <DIV id=powered>Copyright@2007 www.itzcn.com </DIV></DIV> 
8 <table width="705" height="70" border="0" align="center" cellpadding="0" 
re cellspacing="0"> 
完 <tr> 
E3 <td width="199" align="center" bgcolor="#EFEFEF"><span class="STYLE1"> 友 情 链接 : 
学 <a href="www.itzcn.com">IT 在 中 国 </a></span></td> 
习 <td width="9"” bgcolor="#EFEFEF">&nbsp;</td> 
天 <td width="497" align="center" bgcolor="#EFEFEF"><font color="#b3b3b3"> 版 权 所 
册 有 gcopy;IT 在 中 国 [www.itzcn.net]</font> </td> 
</tr> 
</table> 
</BODY> 


将 上 述 代码 保存 ， 文 件 名 为 index.php。 打 开 正 浏览 器 ， 在 地 址 栏 中 输入 http://localhost: 
8080/yan/index.php， 单 击 【 转 到 】 按 钮 ， 会 显示 如 图 17-5 所 示 的 窗口 。 


留言 板 管理 


碍 秋 了 ` 必 给 入 下 面 的 姓名 ) 
晶 除 留言 二 


17-5 留言 板 首页 


通过 留言 板 首页 可 以 实现 查看 全 部 留言 、 删 除 留言 、 添 加 留言 、 查 看 留言 等 功能 。 


17.3 添加 留言 页 面 


在 本 留言 板 中 添加 留言 页 面 作为 留言 板 首页 ， 故 本 节 重 点 介绍 在 数据 库 中 添加 留言 。 下 面 创建 
一 个 PHP 页 面 ， 用 来 实现 提交 的 留言 信息 ， 打 开 记事 本 ， 输 入 下 列 代码 : 


<META http-equiv=Content-Type content="text/html; charset=gb2312"> 
<table width="88%" height="98" border="0" align="center"> 
<tr> 
<td colspan=2 align="center"><embed src="CSS/bt.swf" width="721" 
height="91"></td> 
</tr> 
</table> 
<?php 
require once "Conn.php"; 
$auto=$ POST['autoLogin']; 
$name=$_POST['author']; 
$title=$ POST['subject']; 
$content=$_POST['body']; 
if($auto){ 
if($name=="" or $title=="" or $content==""){ 
echo "<script Language="'JavaScript"'>window.location ="'error. 
php'</script>"; 
} 
$sql="insert into yan (tie name,tie title,tie content) values ('$name', 
"otitie, "Scontent’)™? 
$conn=new Odbcon(); 
$conn->inser ($sql); 


echo "<p align=center> 留 言 成 功 提交 </p>"; 

echo "<script Language='JavaScript'>window.alert (' 留 言 成 功 提交 ， 即 将 返回 留言 页 
面 ')</script>"; 

echo "<script Language="'JavaScript"'>window.location ='index.php'</script>"; 


elsef 
if ($name=="") { 
Stem=rand (); 
$name=" 访 客 ".$tem; 
} 
if($title=="" or $content==""){ 
echo "<script Language='JavaScript'>window.location = 
"error.php'</script>"; 
1 
$sql="insert into yan (tie name,tie title,tie content) values 
('$name','S$title','S$content')"; 
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Sconn=new Odbcon(); 
$conn->inser ($sql); 
echo "<p align=center> 留 言 成 功 提交 </p>"; 
echo "<script Language='JavaScript'>window.alert(' 和 留言 成 功 提交 ， 即 将 返回 留言 页 
面 ')</script>"; 
echo "<script Language="'JavaScript'>window.location ="'index.php'</script>"; 
上 
二 
<table width="723"” height="70" border="0" align="center" cellpadding="0" 
cellspacing="0"> 
站 
<td width="206" align="center" bgcolor="#EFEFEF"><span class="STYLE1"> 友 情 链 接 ， 


将 上 述 代码 保存 ， 文 件 名 为 success.php。 
在 图 17-5 中 ， 有 两 种 留言 的 提交 方式 ， 一 种 是 署名 提交 ， 一 种 是 匿名 提交 。 任 意 选择 其 中 的 一 
种 形式 ， 在 文本 框 中 输入 下 列 信息 ， 如 图 17-6 所 示 。 
加 留言 板 -一 *houyangl2 - Wicrosoft Internet Explorer 


文件 外 编 甸 于 ) 查理 WW 收 宗 旬 。 工具 CD) 帮助 m 
地 小 四 | 接 http://1ocnlhbost 0000/yevisdes phy 


型 <a href="www.itzcn.com">IT 在 中 国 </a></span></td> 

时 <td width="4"” bgcolor="#EFEFEF"><img src="images/kuan.gif" width="4" 
[= height="33"></td> 

了 <td width="604" align="center" bgcolor="#EFEFEF"><font color="#b3b3b3"> 版 权 所 
如 有 gcopy;IT 在 中 国 [www.itzcn.net]</font> 

完 </td> 

htr> 

学 

二 站 </table> 

习 

手 

册 


---- 请 及 时 提出 宝 
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CETTE 


图 17-6 ”提交 留言 


【提交 】 按 钮 ， 会 显示 如 图 17-7 所 示 的 窗口 。 
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ft Internet Erplorer 医 | 


企 鲤 言 成 功 提交 ， 即 枯 返 回 入 言 页 面 


图 17-7 添加 留言 成 功 


在 图 17-6 中 ， 如 果 选 中 【署名 发 表 】 复 选 框 ， 则 必须 在 【姓名 】 文 本 框 中 输入 名 称 。 【主题 】 
文本 框 和 【内 容 】 文 本 框 不 能 为 空 。 

在 上 述 代码 中 ， 使 用 “require_once "Conn.php"” 语 句 把 Conn.php 文件 包含 在 当前 文件 中 ， 并 
使 用 $_POST 方式 获取 客户 端 提交 的 留言 信息 。 如 果 留 言 人 采用 的 是 署名 发 表 ， 则 需要 使 用 站 语句 
对 留言 人 的 姓名 、 主 题 、 内 容 进 行 判 断 ， 三 者 均 不 能 为 空 。 创 建 “insert into yan 
(tie_name,tie_title,tie_content) values ('$name','$title','$content' ) ”语句 实现 将 留言 信息 插入 数据 库 中 ， 
调用 Conn.php 文件 中 的 类 Odbcon 实例 化 一 个 对 象 conn， 并 使 用 对 象 中 的 方法 inser0 执 行 SQL 语 
句 。 如 果 留 言 提交 成 功 ， 则 会 使 用 JavaScript 语言 弹出 一 个 对 话 框 信息 ， ee 
首页 。 如 果 留 言 人 采用 的 是 匿名 发 表 ， 则 只 对 留言 主题 和 留言 内 容 进 行 判断 ， 二 者 均 不 能 为 空 
时 留言 人 的 姓名 将 会 由 随机 产生 的 数字 和 指定 字符 串 组 成 。 下 面 的 i ap Pt 


留言 添加 完成 后 ， 有 时 需要 查看 添加 的 留言 ， 以 进行 确认 。 下 面 创 建 一 个 页 面 实现 该 功能 ， 显 
示 刚 刚 添加 的 留言 信息 。 打 开 记事 本 ， 输 入 下 列 代码 ; 


<table width="88%" height="98" border="0" align="center"> 
<tr> 
<td colspan=2 align="center"><embed src="CSS/bt.swf" width="831" 
height="81"></td> 
</tr> 
</table> 
<table border=0 width=75% align=center> 
<caption><font size=5> 留 言 板 记录 </font></caption> 
<th> 留 言 作者 </th><th> 留 言 题目 </th><th> 留 言 内容 </th> 
<?php 
S$odbcDsn="liuyan™; 
S$odbcUser=""; 
$odbcPass=""; 
$conn=odbc_connect ($0odbcDsn, S$odbcUser, $odbcPass); 


$sql="select top 1 + from yan order by tie id DESC"; 

$result id=odbc do($conn,$sql); 

while(odbc fetch row($result id)) 

{ 

$AAl1=odbc_result ($result id, 2); 
SRAA2=odbc result ($result id, 3); 
SRAR3=odbc result ($result id, 4); 
echo "<tr><td colspan=3><hr size="1" noshade='noshade' 
Color="#000000' styJe= "border-bottom-style:dotted" 
width=800></td></tr>"; 
echo "<tr align=center><td align=center>".S$AAl."</td><td 
align=left>".$AA2."</td><td align=left>".$AA3."</td></tr>"; 


echo "<tr><td colspan=3><hr size='1' noshade='noshade' color= "pear'" 
style='border-bottom-style:dotted' width=800></td></tr>"; 
} 
odbc close($conn); 
pe 
</table> 


<a href="index.php"> 返 回首 页 </a> 
<table width="841" height="70" border="0" align="center" cellpadding="0" 
cellspacing="0"> 
< 
<td width="310" align="center" bgcolor="#EFEFEF"><span class="STYLE1"> 友 情 链接 : 
<a href="www.itzcn.com">IT 在 中 国 </a></span></td> 
<td width="531"” align="center" bgcolor="#EFEFEF"><font color="#b3b3b3"> 版 权 所 
有 &copy;IT 在 中 国 [www.itzcn.net]</font> </td> 
</tr> 
</table> 


将 上 述 代码 保存 ， 文 件 名 为 chajsp。 单 击 图 17-5 中 的 【查看 留言 】 超 级 链接 ， 会 把 程序 的 控制 
权 转 到 chajsp， 该 页 面 的 执行 结果 如 图 17-8 所 示 。 
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板 记录 


留 留言 内 容 
坚决 支持 环境 保护 政策 


返回 首页 


友情 链接 : IT 在 中 国 


17-8 ”显示 留言 信息 


在 上 述 代码 中 ,创建 数据 库 连 接 conn， 并 使 用 函数 odbc do0 获 取 记 录 集 对 象 result id， 该 记录 
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集中 保存 的 信息 是 最 新 添加 的 留言 信息 。 在 while 循环 中 , 使 用 函数 odbc_fetch row0 和 odbc_ result0 
把 记录 集中 的 数据 显示 出 来 。 


17.5 ”显示 全 部 留言 页 面 


通过 上 面 的 添加 操作 ,我 们 知道 所 有 的 留言 信息 都 是 保存 在 Access 数据 库 中 的 yan 表 中 ， 下 面 
把 留言 内 容 一 次 性 显示 出 来 ， 以 供 参考 。 如 果 要 显示 留言 板 中 的 全 部 信息 ， 不 可 避免 地 要 进行 分 页 
显示 。 打 开 记 事 本 ， 输 入 下 列 代码 : 


<table width="88%" height="98" border="0" align="center"> 
<tr> 
<td colspan=2 align="center" bordercolor=pear><embed src="CSS/bt.swf" 
width="904" height="91"></td> 
</tr> 
</table> 
<h3 align=center> 留 言 板 记 录 显 示 </h3> 
<table border=]1 align=center width=81%> 
<caption></caption> 
<th width="21%"> 留 言 序号 </th> 
<th width="21%"> 留 言 人 </th> 
<th width="21%"> 留 言 主题 </th> 
<th width="37%"> 留 言 内 容 </th> 
<?php 


Spagesize = 5; 
$connection=odbc connect('liuyan','',''); 
$query="'select count(*) from yan'; 


$result=odbc do($connection, $query); 
$recordcount=odbc_ result ($result,1); 
odbc free result ($result); 
Spagecount = bcdiv($recordcount+$pagesize-1,$pagesize,0); 
$page=$ GET['page']; 
if($page<1) Spage=17 
if($page>$pagecount) S$page=$pagecount; 
if($page>0) 
echo '<a href="'.$PHP SELF.'?page=1"> 第 一 页 </a> '; 
if ($Spage>1) 
echo '<a href=" '.$PHP SELF.'?page='.($page-1).' "> 上 一 页 
</a>'; 
3 
else{ 
echo ' 前 页 '; 
} 
if ($page<$pagecount) 
{ 
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PHP+MySQL 仆人 绅 镍 尺 炸 率 


2 
</table> 
<table width="800" height="70" border="0" align="center" cellpadding="0" 
cellspacing="0"> 
<tr> 
<td width="273" align="center" bgcolor="#EFEFEF"><span class="STYLE1"> 友 情 链接 : . 
<a href="www.itzcn.com">IT 在 中 国 </a></span></td> 
<td width="4" bgcolor="#EFEFEF"><img src="images/kuan.gif" width="4" 
height="33"></td> 
<td width="523" align="center" bgcolor="#EFEFEF"><font color="#b3b3b3"> 版 权 所 
有 gcopy;IT 在 中 国 [www.itzcn.net]</font> </td> 
</tr> 
</table> 


将 上 述 代码 保存 ， 文 件 名 为 Example.php。 单 击 图 17-5 中 的 【查看 全 部 留言 】 超 级 连接 ， 会 显 
示 如 图 17-9 所 示 的 窗口 。 
镜 http://localhost: 8080/yan/Example, php 


文件 四 遍 汝 区 | 查看) 收藏 公 ) IAV _ WW 
wtu 赠 


留言 板 记录 显示 
第 一 页 前 页 下 一 页 最 后 一 页 每 页 5 条 第 1 页 共 2 页 留言 板 首页 
留言 序号 留言 人 留言 主题 留言 内 容 
一 个 小 知识 分 学 习 雷 笑 好 榜样 ， 缕 续 沈 
81 “明天 前 在 。 你 好 有 。50 年 没有 
区 见面 
84 出海 松 韩国 人 质 祝福 他 们 得 到 释放 
春天 来 了 ， 寒 冷 的 冬天 悄 
87 一 个 小 兵 春天 来 了 ee 万 紫 千 红 的 
天 
88 访客 19300 [色彩 色彩 数 就 是 屏幕 
89 画家 色彩 色彩 


友情 链接 : IT 在 中 国 


ETE 
图 17-9 查看 全 部 留言 


在 上 述 代码 中 ， 主 要 实现 分 页 显示 的 功能 。 其 步骤 和 其 他 实现 分 页 显示 的 功能 基本 相同 ， 首 先 
设 定 每 页 显示 的 记录 数 并 获取 数据 库 中 总 的 记录 数 ， 在 程序 中 使 用 超级 链接 传递 显示 的 页 数 。 


17.6 ”删除 留言 


网 络 上 的 留言 有 各 种 形式 ， 对 于 无 关 主 题 的 留言 一 般 都 要 进行 删除 。 对 于 留言 的 删除 功能 ， 不 
可 能 每 个 人 都 需要 具备 ， 只 能 管理 员 才 能 实现 删除 功能 。 本 留言 板 的 删除 功能 需要 通过 姓名 和 密码 
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的 校 验 。 
1. 管理 员 验 证 
下 面 创建 一 个 页 面 实现 对 留言 板 管理 员 的 校 验 功能 ， 打 开 记 事 本 ， 输 入 下 列 代码 : 


<table width="88%" height="98" border="0" align="center"> 
<tr> 
<td colspan=2 align="center"><embed src="CSS/bt.swf" width="831" 
height="81"></td> 
</tr> 
</table> 
<center> 
<h3 > 请 输入 管理 员 密 码 </h3> 
<form action=del00.php method=post> 
管理 员 名 称 : <input type=text name=namel><br> 
管理 员 密 码 : <input type=text name=name2><Br> 
<input type=submit value= 校 验 ><input type=reset value= 重 写 > 
</form> 
</center> 
<a href="index.php"> 返 回首 页 </a> 
<table width="841"” height="70" border="0" align="center" cellpadding="0" 
cellspacing="0"> 
<tr> 
<td width="310" align="center" bgcolor="#EFEFEF"><span class="STYLE1"> 友 情 链接 : 
<a href="www.itzcn.com">IT 在 中 国 </a></span></td> 
<td width="531" align="center" bgcolor="#EFEFEF"><font color="#b3b3b3"> 版 权 所 
有 gcopy;IT 在 中 国 [www.itzcn.net]</font> </td> 
</tr> 
</table> 


将 上 述 代 码 保存 ， 文 件 名 为 del0.php。 单 击 图 17-5 中 的 【删除 留言 】 超 级 链接 ， 程 序 的 控制 权 
会 转向 del0.php 页 面 ， 执 行 结果 如 图 17-10 所 示 。 


oft Intornot Exp 


是 
a 
卫 
十 
三 
< 
CD 
名 
T= 
| 一: 
TU 
多 
到 
习 
手 
册 


文件 四 编 骆 G@) 查看 中 荣昌 工具 CD) 帮助 0 
地 站 四 )| 罩 nte /leeslhost:acB0/yar/aol0 phy 


请 输入 管理 员 密 码 


返回 首页 
友情 链接 : IT 在 中 国 


图 17-10 管理 员 验 证 页 面 
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由 


从 上 述 代 码 中 ， 本 页 面 只 是 校 验 功能 的 显示 页 面 ， 如 果 在 图 17-10 中 输入 信息 ， 会 把 该 信息 提 
交 给 校 验 页 面 。 下 面 创 建 信息 的 校 验 页 面 ， 打 开 记事 本 ， 输 入 下 列 代码 : 


<?php 
$name=$ POST['namel']; 
$pass=$_POST['name2']; 
if ($name=="" | $pass==""){ 
echo "<script Language='dJavaScript'>window.alert(' 名 称 密码 不 能 为 空 ， 即 将 返回 首 
页 ')</script>"; 
echo "<script Language="'JavaScript"'>window.location ='index.php'</script>"; 
} 
if($name=="'admin' & $pass=="'admin'){ 
echo "<script Language='JavaScript'>window.alert (' 名 称 和 密码 正确 ， 即将 进入 删除 
页 面 ')</script>"; 
echo "<script Language="'JavaScript'>window.location ='xian.php'</script>"; 
} 
else 


让 
echo "<script Language='JavaScript'>window.alert (' 名 称 和 密码 不 正确 ， 即 将 返回 管 


理 员 页 面 ') </script>"; 
echo "<script Language='JavaScript'>window.location ='del0.php'</script>"; 


} 


tt 
将 上 述 代 码 保存 ， 文 件 名 为 del00.php。 该 页 面 没 有 相应 的 显示 页 面 ， 只 是 完成 控制 功能 。 如 果 
提交 的 信息 为 室 ， 则 返回 指定 页 面 。 如 果 提 交 的 管理 员 名 称 为 admin， me rr 国 


并 且 管 理 员 密码 为 admin， 则 转向 留言 板 的 删除 页 面 ， 否 则 重新 返回 校 和 
验 页 面 。 CD 


在 图 17-10 中 ， 分 别 在 【管理 员 名 称 】 文 本 框 和 【管理 员 密码 】 文 


本 框 中 输入 admin。 输 入 完毕 后 ， 单 击 【 校 验 】 按钮， 会 显示 如 图 17-11 。” 图 17-11 提示 对 话 框 
所 示 的 对 话 框 。 
2.， 删除 留言 


下 面 创 建 删除 留言 的 显示 页 面 ， 打 开 记事 本 ， 输 入 下 列 代码 ; 


<table width="88%" height="98" border="0" align="center"> 
<tr> 
<td colspan=2 align="center"><embed src="CSS/bt.swf" width="660" 
height="110"></td> 
</tr> 
</table> 
<h3 align=center> 删 除 留言 页 面 </h3> 
<table border=1 align=center width=68%> 
<th width="27g%"> 留 言 人 </th> 
<th width="38%"> 留 言 题 目 </th> 
<th width="35%"> 操 作 </th> 
<?php 


$odbcDsn="liuyan™; 


3 溃 


这 
留 
板 
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$odbcUser=""; 

$odbcpass=""; 

$conn=odbc connect($odbcDsn, $odbcUser， $odbcPass); 
$sql="select * from yan "7 

$result id=odbc dol($conn,$sql); 

while(odbc fetch row($result id)) 


{ 
$AA0 = odbc result($result id, 1); 
$AAl = odbc result(Sresult id, 2); 
$AA2 = odbc result(Sresult id, 3); 


echo "<form action='del.php' method=post> 
<tr><td>".$AAl."</td><td>".$AA2."<td><input type=submit value= 删 除 ><input 
type=hidden name=namel value=".$AA0."></td></tr></form>"; 
} 
odbc close($conn); 
?> 
</table> 
<center><a href="index.php”> 返 回首 页 </a></center> 
<table width="693" height="70" border="0" align="center" cellpadding="0" 
cellspacing="0"> 
<tr> 
<td width="219" align="center" bgcolor="#EFEFEF"><span class="STYLE1"> 友 情 链接 : 
<a href="www.itzcn.com">IT 在 中 国 </a></span></td> 
<td width="474" align="center" bgcolor="#EFEFEF"><font color="#b3b3b3"> 版 权 所 
有 &copy;IT 在 中 国 [www.itzcn.net]</font> </td> 
</tr> 
</table> 


将 上 述 代码 保存 ， 文 件 名 为 xian.php。 单 击 图 17-11 中 的 【确定 】 按 钮 ， 程 序 的 控制 权 会 转向 
页 面 xian.php， 执 行 结果 如 图 17-12 所 示 。 
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立 伞 串 。 编 提 久 查看 访 】 中 来 工具 立 ) 压 有 的) 
Wi | mop /Neont wor pee 


删除 留言 页 面 
留言 人 留言 题目 


一 个 小 知识 分 子 
刘海 松 

一 个 小 兵 

访客 19300 

西 察 

蜗牛 


友情 链 楼 : I 在 中 国 


图 17-12 留言 删除 页 面 
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在 上 述 代码 中 ， 留 言 表 中 的 信息 以 表格 的 形式 全 部 显示 出 来 ， 并 在 每 行 数据 记录 的 后 面 加 入 了 
【删除 】 操 作 。 如 果 要 删除 某 行 记 录 ， 直 接 单 击 【删除 】 按 钮 即 可 。 需 要 注意 的 是 ， 此 处 使 用 了 隐藏 
按钮 来 实现 删除 信息 的 传递 。 下 面 创建 一 个 页 面 ， 实 现 留言 板 中 的 记录 删除 操作 。 打 开 记 事 本 ， 输 
入 下 列 代码 : 


<table width="88%" height="98" border="0" align="center"> 
<tr> 
<td colspan=2 align="center"><embed src="CSS/bt.swf" width="831" 
height="81"></td> 
</tr> 
</table> 
<?php 
require once "Conn.php"; 
$auto=$_POST['namel']; 
$sql="delete from yan where tie id=$auto"; 
$conn=new Odbcon(); 
$conn->inser ($sql); 
echo "<script Language='JavaScript'>window.alert (' 数 据 删除 成 功 ， 即 将 返回 删除 页 面 ' ) 
</3cript>"s 
echo "<script Language="'JavaScript'>window.location ='xian.php'</script>"; 
2 
<table width="841" height="70" border="0" align="center" cellpadding="0" 
cellspacing="0"> 
<tr> 
<td width="310" align="center" bgcolor="#EFEFEF"><span class="STYLE1"> 友 情 链 接 : 
<a href="www.itzcn.com">IT 在 中 国 </a></span></td> 
<td width="531" align="center" bgcolor="#EFEFEF"><font color="#b3b3b3"> 版 权 所 
有 &copy;IT 在 中 国 
[www.itZzcn.net]</font> </td> 
</tr> 
</table> 


将 上 述 代码 保存 ， 文 件 名 为 del.php。 单 击 图 17-12 中 的 【删除 】 按 钮 ， 会 显示 如 图 17-13 所 示 
的 对 话 框 。 


Dhttp://localhost: 3080/yan/del. php - Microsoft Internet Erplorer 


地 址 由) | 阐 http://1ocslhost:8080/yan/del php 


图 17-13 ”删除 成 功 


在 上 述 代码 中 ， 使 用 “require once "Conn.php"” 语 句 将 Conn php 文件 包含 在 当前 文件 中 ， 然 
后 使 用 $_POST 获取 xian.php 页 面 传递 过 来 的 删除 信息 ， 并 调用 Conn .php 文件 中 的 类 Odbcon 实例 
化 一 个 对 象 conn， 使 用 该 对 象 实现 删除 功能 。 


417 


第 18 章 会 员 管理 系统 


学 习 目 标 |oujecuve 


在 网 络 上 ， 基 本 上 每 一 个 专业 网 站 都 有 自己 的 会 员 〈 网 站 的 注册 用 户 ) 管理 系统 。 会 员 是 网 站 
发 展 的 核心 ， 网 站 的 许多 服务 都 是 为 网 站 会 员 提供 的 ， 所 以 方便 有 效 地 管理 网 站 会 员 ， 对 于 一 个 专 
业 网 站 来 说 是 至 关 重 要 的 工作 。 本 章 主 要 介绍 如 何 通过 PHP+MySQL 来 完成 会 员 管理 的 工作 ， 包 括 
会 员 申请 、 会 员 登录 、 会 员 信息 修改 、 会 员 信息 搜索 、 管 理 员 对 会 员 进 行 管理 等 。 


内 容 摘要 1awcwoct 


了 解 会 员 管 理 系统 的 实现 功能 
掌握 基于 MySQL 的 用 户 注册 

掌握 基于 MySQL 的 会 员 查询 

掌握 基于 MySQL 的 会 员 登录 与 注销 
掌握 如 何 进行 会 员 信息 修改 

理解 并 掌握 管理 员 对 会 员 的 管理 
掌握 如 何 进行 删除 会 员 操作 
理解 会 员 管理 系统 的 设计 理念 


18.1 系统 整体 设计 


通常 情况 下 ， 在 一 个 虚拟 的 网 络 社区 中 ， 其 最 基本 的 模块 是 用 户 的 注册 、 登 录 以 及 管理 模块 。 
会 员 管 理 系 统 虽然 只 是 整个 网 站 的 一 小 部 分 ， 但 却 是 很 重要 的 一 个 模块 。 因 为 通常 情况 下 每 一 个 网 
站 都 有 其 自己 的 用 户 群 ， 而 网 站 的 很 多 功能 也 只 对 其 注册 用 户 ( 会 员 ) 开放 ， 因 此 处 理 好 用 户 的 注 
册 、 登 录 及 管理 是 运行 网 站 的 一 个 基础 。 要 实现 该 会 员 管理 系统 ， 首 先 必 须 创建 符合 网 站 要 求 的 用 
户 信息 表 、 管 理 员 信息 表 ， 其 次 是 要 解决 用 户 注册 的 问题 ， 接 下 来 是 如 何 实现 注册 用 户 的 登录 及 自 
我 管理 的 问题 ， 最 后 还 要 涉及 到 管理 员 对 注册 用 户 的 管理 以 及 用 户 信息 的 搜索 、 查 看 等 问题 。 关 于 
用 户 登 录 ， 如 今 网 站 可 以 使 用 Cookie 或 Session 来 实现 ， 这 里 选择 使 用 Session 。 

在 本 章 的 案例 中 ， 开 发 一 个 名 叫 “ 中 软 大 学 强人 网 络 协会 ”的 会 员 管理 系统 。 该 协会 是 在 校 大 
学 生 组 织 而 成 的 团体 ， 它 用 于 为 喜欢 计算 机 网 络 编程 的 在 校 学 生 提 供与 其 他 同学 进行 交流 的 机 会 。 
而 该 会 员 管理 系统 是 进行 会 员 管理 ， 展 示 会 员 风 采 的 一 个 平台 。 在 功能 方面 ， 该 会 员 管 理 系统 基本 
上 实现 了 通用 会 员 系统 的 基本 功能 ， 读 者 通过 稍 加 修改 ， 便 能 把 该 系统 应 用 到 自己 的 网 站 中 。 它 主 
要 实现 以 下 功能 : 

e。 用 户 可 以 申请 加 入 该 协会 。 


VvVvvvyvyveyvyv 


e 如 果 用 户 加 入 协会 的 申请 通过 ， 便 可 进行 登录 。 
。 成 功 登录 后 ， 用 户 可 以 管理 自己 的 信息 或 进入 会 员 中 心 执行 更 多 的 操作 ,以 及 决定 是 否 退 出 


该 协会 。 . 
。 未 加 入 协会 的 用 户 可 以 浏览 所 有 会 员 的 简介 或 搜索 某 个 或 菜 些 会 员 , 但 不 可 以 查看 会 员 的 详 : 
细 信 息 。 a 


。 首页 显示 管理 员 推 荐 的 会 员 、 最 新 加 入 的 会 员 及 会 员 人 员 统 计 等 。 

管理 员 (通常 情况 下 ， 由 协会 负责 人 担任 ) 可 以 批准 用 户 加 入 协会 、 推 荐 会 员 在 本 系统 首页 
显示 、 删 除 会 员 以 及 提升 会 员 为 管理 员 等 。 

。 对 会 员 的 访问 次 数 进行 统计 ， 并 写 入 数据 库 中 。 

。 实现 用 户头 像 图 片 的 上 传 及 分 页 显示 功能 。 


18.2 ”数据 库 设计 


会 员 管理 系统 是 一 个 网 站 最 基本 的 模块 ， 而 会 员 管理 系统 的 数据 表 则 是 这 个 模块 的 基础 ， 因 此 
在 实现 会 员 管理 系统 时 ， 必 须 首先 考虑 如 何 合理 设计 和 安排 会 员 管理 数据 表 的 结构 。 下 面 介 绍 该 会 
员 管理 系统 数据 表 的 创建 过 程 。 

首先 在 MySQL 数据 库 中 创建 一 个 数据 库 member， 用 于 存放 会 员 管理 系统 的 所 有 数据 。 然 后 创 
建 一 个 用 户 信息 表 user， 用 于 存放 用 户 的 基本 信息 ， 如 姓名 、 人 性别 、 年 龄 、 专 业 及 系 别 等 。 最 后 ， 
创建 一 个 管理 员 信息 表 admin， 用 于 存放 管理 员 的 相关 信息 ， 如 用 户 名 、 密 码 等 。 下 面 分 别 对 这 些 
表 及 其 创建 进行 介绍 。 

1. 用户 信息 表 

该 表 主 要 用 来 保存 用 户 的 登录 名 称 〈 会 员 ID)、 真 实 姓名 、 密 码 、 电 子 邮 件 、 联 系 电话 和 通信 
地 址 等 会 员 信 息 。 该 表 中 的 用 户 只 有 在 登录 成 功 后 才 可 以 修改 自己 的 会 员 信息 、 查 看 其 他 会 员 的 详 
细 信 息 以 及 进入 用 户 中 心 执行 其 他 会 员 特 有 的 操作 。 该 表 的 表 名 为 user， 它 的 具体 说 明 如 表 18-1 
所 示 。 
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表 18-1 用 户 信息 表 

字段 名 备注 
1 lm) | 百 | 自动 篇 号 
Userid [vachallo0) | | 人 员 D 
Checkflag |vacha(5) | 是 | 是 否 通过 申请 
Name jvacha) | 否 | 真实 姓名 
Pass 会 员 密码 
Sex varchar(4) 会 员 性 别 
Email varchar(100) 电子 邮件 
Oicq varchar(10) OICQ 号 码 
Userfrom varchar(10) 来 自 ( 省 会 名 ) 


City varchar(14) 来 自 ( 城 市 ) 
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字段 名 备注 

Address varchar(80) 是 通信 地 址 
Zip varchar(6) 是 邮件 编码 
Tel varchar(14) 是 联系 电话 
Edu varchar(10) 是 目前 学 历 
Hangye varchar(20) 是 目前 职业 
Zhuanye varchar(20) 是 所 在 院 系 
Class varchar(4) 是 年 级 
Techang varchar(150) 是 爱好 特长 
Intro varchar(250) 是 个 人 简介 
Regyear varchar(12) 是 入 会 时 间 
Birthday date 是 生日 
Regtime datetime 是 注册 时 间 
Photo varchar(100) 是 个 性 图 像 
Regip Varchar(100) 是 注册 时 的 人 P 
Hit int(10) 默认 值 0 单 击 次 数 
Weekhit int(10) 默认 值 0 周 单 击 次 数 
Best varchar(5) 是 是 否 推 荐 


在 数据 表 user 中 ， 字 段 id 用 于 标识 每 条 记录 ; 字段 userid 用 于 记录 用 户 注册 的 用 户 ID， 由 于 
用 户 D 是 用 户 之 间 相 互 区 别 的 一 个 重要 标志 ， 因 此 用 户 ZD 不 能 相同 ; 字段 photo 用 于 存放 用 户 个 
性 图 片 的 路 径 ， 而 该 图 片 是 通过 上 传 组 件 传 到 服务 器 的 相应 位 置 的 ， 其 他 字段 都 是 用 来 存储 注册 用 
户 各 方面 的 基本 信息 ， 表 中 已 经 进行 了 说 明 。 

2. 管理 员 信息 表 

该 表 主要 用 来 保存 管理 员 的 登录 名 称 〈 管 理 员 ID)、 密 码 、 提 升 为 管理 的 时 间 等 。 该 表 中 的 用 
户 只 有 在 登录 成 功 后 才 可 以 修改 自己 的 会 员 信息 、 查 看 其 他 会 员 的 详细 信息 并 进行 管理 ， 如 删除 现 
有 会 员 、 推 荐 会 员 等 。 该 表 的 表 名 为 admin， 它 的 具体 说 明 如 表 18-2 所 示 。 

表 18-2 ”管理 员 信息 表 


字段 名 备注 
Admin 管理 员 ID 
Adminpwd varchar(10) 管理 员 密 码 
Uptime datetime 提升 为 管理 员 的 时 间 


3. 数据 表 的 创建 

下 面 按照 前 面 表 的 结构 在 MySQL 中 创建 相关 的 信息 表 。 首 先 创建 数据 库 member， 然 后 使 用 
member 为 当前 数据 库 创建 管理 员 信息 表 admin， 具 体 过 程 如 下 所 示 : 

mysql> CREATE DATABASE member2; 

Query OK, 1 row affected (0.03 sec) 


mysql> USE member2; 
Database changed 
mysql> CREATE TABLE admin( 
-> admin varchar(10) NOT NULL, 
-> adminpwd Varchar (10) NOT NULL, 
-> uptime datetime, 
-> primary key(admin)); 
Query OK, 0 rows affected (0.14 sec) 


接着 按照 上 面 创建 管理 员 信息 表 的 方法 创建 用 户 信 息 表 user。 当 然 上 面 只 是 一 种 方法 ， 用 户 还 
可 以 使 用 phpMyAdmin 或 其 他 操作 MySQL 的 工具 进行 创建 。 读 者 也 可 以 通过 将 上 述 操作 过 程 中 的 
SQL 语句 提取 出 来 ， 通 过 和 PHP 程序 相 结合 以 运行 Web 页 面 方式 进行 创建 。 但 是 ， 使 用 MySQL 
客户 端 创建 是 最 基本 的 方法 ， 读 者 一 定 要 认真 掌握 。 
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通常 情况 下 , 网 站 将 用 户 分 为 几 种 类 型 ,比如 以 IT 在 中 国 技术 论坛 (www.itzcn.com/bbs) 来 说 ， 
首先 是 没有 注册 的 普通 用 户 ， 只 享有 在 网 站 上 获取 基本 信息 的 权利 〈 如 查看 别人 发 表 的 主题 信息 )， 
但 是 不 能 享有 一 些 特殊 服务 ， 比 如 发 表 新 帖 ， 下 载 附件 等 ， 然 后 是 已 经 注册 的 用 户 ， 他 们 却 享有 这 
些 权 利 ， 另 外 还 有 版 主 、 管 理 员 等 ， 他 们 对 普通 用 户 及 其 发 表 的 主题 信息 有 管理 的 权利 。 


18.3.1 注册 页 面 


在 本 案例 中 注册 模块 是 申请 成 为 协会 会 员 的 过 程 ， 其 中 加 入 协会 页 面 也 是 用 户 注册 页 面 ， 它 用 
于 将 收集 到 的 用 户 个 人 信息 提交 给 regok.php 页 面 进行 处 理 。 当 然 ， 由 于 协会 性 质 的 不 同 ， 这 里 要 
求 申请 用 户 应 提供 详细 的 信息 。 它 的 具体 代码 如 下 所 示 : 


<?php 
session start(); 
require "conn.php"; 
区 二 
<meta http-equiv="content-type" content="text/html; charset=gb2312" /> 
<title><?php echo $myschool;?> <?php echo S$orgname; ?> 会 员 管 理 系 统 </title> 
<link href="default.css" rel="stylesheet" type="text/css" /> 
<?php 
// 包 含 进来 各 个 模块 
require "top.php"; 
require "regframe.php"; 
require "sidebar.php"; 
require "foot.inc"; 
> 
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看 了 上 面 的 案例 ， 读 者 也 许 会 发 现 这 里 没有 任何 的 有 关 让 用 户 输入 信息 的 代码 ， 因 为 这 里 我 们 
把 各 个 通用 部 分 放 在 了 一 个 文件 中 ， 所 以 本 案例 的 大 部 分 主页 面 都 包含 top.php、sidebarphp 和 
foot.inc 这 三 个 模块 ， 关 于 它们 的 详细 内 容 将 在 后 面 介绍 。 注 册页 面 最 重要 的 文件 是 regframe.php。 
它 的 部 分 代码 如 下 所 示 : 


<input class=pl name=bl type=submit value=" 确 定 " style="color: #000000; background— 
Color: #f3f3f3; border-style: solid; border-width: 1" onmouseover ="this.style. 
backgroundcolor="#ffffff'" onmouseout ="this.style.backgroundcolor="#f3f3f3'"> 
<input class=pl name=b2 type=reset value=" 清 除 " style="color: #000000; background— 
color: #f£f3f3f3; border-style: solid; border-width: 1" onmouseover ="this.style. 
backgroundcolor="#ffffff'" onmouseout ="this.style.backgroundcolor="'#f3f3f3'"> 
</td> 
</tr> 
</tbody> 
</table> 
</center> 
</div> 
</form> 
</td> 
</tr> 
</table> 


</1i> 
</ol> 
</div> 
</div> 


18.3.2 ”注册 处 理 页 面 


该 注册 处 理 页 面 的 实现 文件 是 regok.php， 它 用 于 将 由 reg.php 页 面 传递 过 来 的 用 户 信息 插入 表 
user 中 。 当 然 ， 在 插入 记录 之 前 进行 了 检查 ， 如 果 存 在 相同 的 用 户 ID、 某 些 数据 格式 不 正确 或 存在 
不 允许 为 空 的 项 都 将 导致 插入 失败 ， 并 给 出 相应 的 提示 。 该 页 面 的 具体 代码 如 下 所 示 : 


<?php 

require "conn.php"; 

require "top.php"; 

?> 

<meta http-equiv="Content-Type" content="text/html; charset=gb2312" /> 
<title><?php echo $myschool;?>_ <?php echo $orgname;?> 加 入 协会 执行 结果 </title> 
<link href="default.css" rel="stylesheet" type="text/css" /> 

<? 

$userid=trim($ POST['userid']); 

$name=trim($ POsST['name']); 

$sex=trim($ POST['sex']); 

$email=trim($ POST['email']); 

S$passwd=trim($_ POST['passwd']); 

Srepasswd=trim(S POST['repasswd']); 

S$oicq=trim($ POST['oicq']); 

$city=trim($ POST['city']); 

$address=trim($ POST['address']); 

$zip=trim($ POST['zip']); 

$tel=trim($ POST['tel']); 


[| 


册 


尝 洲 腊 式 男 六 


卫 
a 
五 
十 
= 

Se 
a] 
[a 
re 
一 
TU 
全 
学 
习 

手 
册 


424 


$userfrom=trim($ POST['userfrom'])7 
$year=trim($ POST['year']); 
$month=trim($ POST['month']); 
$day=trim($ POST['day'"])7 
$zhuanye=trim($ POST['zhuanye']); 
$class=trim($ POST['class']); 
$hangye=trim($ POST['hangye']); 
$techang=trim($ POST['techang']); 
$regyear=trim($ POST['regyear']); 
$intro=nl2br (htmlspecialchars($ POST["'intro'])); 
$regip=$_SERVER['REMOTE ADDR']; 
$birthday=$year."-".S$month."-".$day; 
$regtime = date("Y-m-d H:i:s"); 

$edu = trim($ POsST['edu']); 

$photo = trim($ POST['photo']); 


if(($userid=="") or ($name=="") or ($sex=="") or ($email=="") or ($passwd== 
($repasswd=="") or ($userfrom=="") or ($year=="") or ($month=="") or ($day: 
($city=="") or ($address=="") or ($zip=="") or ($tel=="") or ($techang=="") or 
($zhuanye=="") or ($class== 


{ 


echo "<script language=javascript>alert (' 带 *# 号 的 选项 必须 填写 全 ! ') ;history. 
go(-1)</script>"; 
exit; 
} 
if(substr count ("$userid", " ")>0 or substr count ("$userid", " ")>0 or 
substr count ("$name", " ")>0 or substr count ("$userid", " ")>0) 
1 
echo "<script language=javascript>alert(" 账 号 和 姓名 不 能 有 空格 ') ;history.go(-1) 


x acripe>™y 
exit; 
上 
if (!ereg("^[ a-z0-9-]",$userid)) 
:| 
echo "<script language=javascript>alert(' 您 的 账号 格式 不 对 ， 只 能 是 英文 或 者 数字 ' ) 
history.go(-1)</script>"; 
exit; 
} 


if (!ereg("^[0-9]",$year)) 
1 
echo "<script language=javascript>alert(' 生 日 年 份 只 能 用 数字 表示 ' ) ;history. 
go(-1)</script>"s 
exit; 
$sql="SELECT userid FROM $user where userid="'S$userid'"; 
$result=mysql query($sq]l, $db); 
if($myrow=mysql fetch row($result)) 
| 
echo "<script language=javascript>alert('" 此 账号 有 人 注册 ， 请 重新 填写 ! ' ) ;history. 


go(-1)</script>"; 


exit; 
if($passwd<>$repasswd) 
1 


echo "<script language=javascript>alert (' 两 次 输入 的 密码 不 一 样 ， 请 检查 ') ;history. 
go(-T)</script>"r 
exit; 

} 

if (Vereg(""“[ a-z0-9-]+(N\.[ a=z0=9=]+})*@ [a-20-9-]+(\=[a-z0-9-]+)*9"; $emall)}) 

jl 
echo "<script language=javascript>alert(' 您 的 Email 地 址 格式 不 对 ， 请 检查 ') ; 
history.go(-1)</script>"; 
exit; 


if (is uploaded file($ FILES['photo']['tmp name'’])) 
1 

copy($ FILES['photo']['tmp name'], 

"photo/".$ FILES['photo'] ['name']); 

1 
else 
tf 

echo "<p> 上 传 出 错 .</p>"; 
| 
$photo name= "Photo/".$_FILES["photo']["name'"]7 


$query="INSERT INTO $user (userid,name,pass, sex,email,oicq,userfrom,city,address, 
7 
."tel,edu,hangye, zhuanye, class, techang, intro, regyear,birthday, 
regtime,photo, regip)" 
. "VALUES (' $userid','S$name','$passwd','$sex','$email','$oicqg', 
'$Suserfrom'," 
."'$city','S$address','$zip','$tel','$edu','$hangye', '$zhuanye', 
WCG Stechang. SLntrou 
."'$regyear', '$birthday', '$regtime','$photo name','S$regip')"; 


$result = mysql query ($query) 
or die ("查询 出 现 错误 : " . mysql] error()); 
$rowid = mysql insert id(); 
if ($rowid>0) 
{ 
echo "加 入 协会 成 功 ! <hr><p>"; 
echo "新 加 入 协会 的 用 户 是 : " .$name."<p>"; 
echo "<br><a href='reg.php'> 返 回首 页 </a>"; 
} 
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{ 
echo "加 入 协会 失败 ! <hr><p>"; 
echo "用 户 : [".$name."]"; 
echo "添加 失败 ! "; 
echo "<br><a href='reg.php'> 重 新 加 入 协会 </a>"; 
} 
2 
<2php 


require "foot.inc"; 

2 

在 上 述 代码 中 ， 首 先 包 含 进来 连接 数据 库 代码 的 文件 conn.php， 以 及 其 他 通用 模块 部 分 ， 其 次 
通过 $ POSTD 获 取 从 reg.php 页 面 传递 过 来 的 数据 ， 接 着 对 获取 的 数据 进行 判断 ， 以 保证 所 提交 的 
数据 符合 要 求 ， 最 后 使 用 建立 的 连接 执行 插入 操作 ， 并 根据 操作 结果 返回 相应 的 信息 。 


18.3.3 ”测试 注册 模块 


把 注册 模块 相关 文件 reg.php 和 regok.php 存放 在 Apache 目录 下 的 htdocs\hy 子 目 录 下 , 打开 正 
浏览 器 ， 在 地 址 栏 中 输入 http://localhost/hy/reg.php， 按 Enter 键 会 显示 如 图 18-1 所 示 的 页 面 。 
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图 18-1 注册 页 面 
在 该 页 面 中 输入 用 户 的 个 人 信息 ， 这 里 输入 用 户 账号 (用户 卫 ) 为 “bob”， 因 为 该 值 为 用 户 的 


标识 ， 是 唯一 的 ， 所 以 当 数据 库 中 已 经 存在 此 用 户 ID 时 ， 系 统 将 给 出 错误 提示 ， 如 图 18-2 所 示 。 
否则 ， 在 保证 其 他 信息 都 符合 要 求 的 情况 下 ， 使 用 用 户 ID 为 “bob2” 便 可 以 注册 成 功 ， 如 图 18-3 


习 申 软 大 学 强人 网 | 


加 入 协会 成 功 1 . 


新 加 入 协会 的 用 户 是 : bob2 


返回 首页 


图 18-2 重复 使 用 用 户 ID 注册 图 18-3 用 户 注册 成 功 
这 里 需要 注意 的 是 ， 在 上 述 代码 对 获取 数据 进行 检测 时 使 用 了 正则 表达 式 ， 如 下 所 示 : 


if (!ereg("^[ a-z0-9-]+(\.[_a-z0-9-]+)*@[a-z0-9-]+(\. [a-z0-9-]+)*$", $email)) 
{ 


互 站 


册 


echo "<script language=javascript>alert(' 您 的 Email 地 址 格式 不 对 ， 请 检查 ') ; 
history.go(-1)</script>"; 
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} 
该 正则 表达 式 通 过 对 用 户 输入 电子 邮件 格式 的 检测 ， 能 确保 用 户 输 入 的 电子 邮件 是 正确 的 。 


18.4 ”查询 模块 


对 一 个 小 型 网 站 来 说 ， 也 许 查 询 模块 并 不 是 太 需 要 ， 但 如 果 针 对 一 个 商业 类 型 的 网 站 来 说 ， 那 
么 查询 模块 是 必 不 可 少 的 ， 因 为 巨大 的 会 员 数 量 将 使 其 管理 难度 增 大 。 在 数 万 到 数 十 万 或 更 多 的 会 
员 中 查找 某 一 个 或 某 些 会 员 ， 如 果 不 使 用 查询 而 一 页 页 地 进行 查找 ， 工 作 量 之 大 是 可 想 而 知 的 。 


18.4.1 查询 页 面 


查询 模块 包括 提交 查询 信息 页 面 和 查询 信息 处 理 页 面 。 其 中 ， 查 询 页 面 seekuserphp 同样 包含 
通用 的 几 个 模块 ， 这 里 重点 介绍 它 所 包含 的 seekuserframe.php， 该 文件 包含 了 获取 查询 信息 的 实现 
代码 。 部 分 代码 如 下 所 示 : 


<div id="content"> 
<div id="main"> 
<div id="welcome"> 
<h2><?php echo $orgname; 2> 会 员 模糊 搜索 </h2> 
人 相让 本 
<form name="forml" method="post" action="searchok.php"> 
<table border="0" cellspacing="0" cellpadding="0"> 
<tr> 
<td>gnbsp; gnbsp; gnbsp; &nbsp; tnbsp; gnbsp; 
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在 查询 页 面 中 ， 用 户 可 以 按照 会 员 的 性 别 、 所 在 院 系 以 及 会 员 的 家 乡 分 别 进 行 模糊 搜索 ， 或 者 
以 它们 的 组 合 来 进行 模糊 搜索 ， 另外， 用 户 还 可 以 按照 会 员 的 真实 姓名 来 进行 精确 的 搜索 。 


18.4.2 ”查询 信息 处 理 页 面 


查询 页 面 seekuserphp 将 数据 提交 给 searchok.php 进行 处 理 , 由 于 searchok.php 页 面 采用 了 与 注 
册页 面 reg.php 同样 的 形式 ， 实 际 上 是 提交 给 了 searchokframe.php 进行 处 理 ， 所 以 查询 信息 处 理 页 
面 的 实现 文件 是 searchokframe .php。 

它 的 具体 代码 如 下 所 示 : 


PHP+MySQL 休 钊 洲 ? 中 宣 


仲 钉 民 慎 柬 


PHP+MySQL 


在 上 述 代码 所 呈现 的 查询 信息 处 理 页 面 中 , 实现 了 PHP 编程 中 较为 复杂 、 但 常用 的 功能 。 主 要 
有 以 下 几 个 部 分 : 

(1) 要 判断 传递 过 来 的 参数 是 用 户 提交 查询 传递 过 来 的 ， 还 是 用 户 在 查询 结果 中 进行 翻 页 传递 
过 来 的 ， 因 为 前 者 需要 使 用 $_POSTI] 方 式 获取 参数 值 ， 而 后 者 则 需要 使 用 $_GETI[] 方 式 获取 参数 值 。 

(2) 要 判断 用 户 以 何 种 方式 进行 查询 ， 如 果 是 精确 查询 ， 则 按照 会 员 的 真实 姓名 进行 查询 很 简 
单 ; 而 模糊 查询 则 不 同 ， 由 于 这 里 是 针对 多 项 进行 的 ， 所 以 它们 有 多 种 组 合 方式 ， 如 按 性 别 、 性 别 
+ 所 在 院 系 、 性 别 + 所 在 院 系 + 会 员 家 乡 等 。 

(3) 对 查询 结果 进行 分 页 ， 并 根据 用 户 当前 的 访问 情况 判断 如 何 显示 分 页 链接 。 如 果 当 前 用 户 
访问 的 是 第 一 页 那么 【第 一 页 】 和 【上 一 页 】 将 没有 任何 链接 ; 如 果 访 问 的 是 最 后 一 页 ， 那么 【下 
一 页 】 和 【 尾 页 】 将 没有 任何 链接 。 
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18.4.3 ”测试 查询 模块 


把 查询 模块 相关 文件 seekuserphp、seekuserframe.php、searchok php 和 searchokframe.php 存放 
在 Apache 目录 下 的 htdocs\hy 
子 目 录 下 ,打开 正 浏览 器 ， 
在 地 址 栏 中 输入 http:// 
localhost/hy/seekuser.php ， 按 强人 网 络 协会 会 员 模 糊 搜索 
Enter 键 会 显示 如 图 18-4 所 示 a 
的 页 面 。 

在 该 页 面 中 ， 选 择 所 在 院 
系 : 计算 机 与 信息 工程 学 院 ， 
然后 单 击 【 搜 索 】 按 钮 ， 如 果 > | 
存在 查询 结果 将 显示 出 来 ， 如 . 强人 网 络 协会 会 员 精 确 搜索 g 
图 18-5 所 示 。 如 果 未 找到 符合 oid 
条 件 的 记录 ， 系 统 将 给 出 类 似 
于 如 下 所 示 的 提示 信息 : 图 18-4 查询 页 面 
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本 站 提示 


对 不 起 ， 没 有 找到 相应 的 记录 ! 你 查找 的 协会 会 员 不 存在 ! 
重新 搜索 


虱 中 软 大 学 强人 网 绍 切 会 _ 会 员 物 索 结 果 - icrosoft Internet Explorer 
袜 件 ) 六 名 ) 查 香 V) 收 基 (工具 TT) 六 盈 00 
出 外 网 | 鸭 http /jlscanhesryhy/searehok ph 

全 


会 会 员 捉 索 结果 如 下 : 


18-5 查询 处 理 结果 页 面 


18.5 ”显示 模块 


显示 模块 包括 两 种 类 型 的 显示 ， 一 种 是 会 员 概 要 信息 显示 页 面 ， 显 示 方 式 类 似 于 图 18-5; 另 一 。。 
种 是 会 员 详细 信息 显示 页 面 ,能 访问 此 页 面 的 用 户 必 天 
须 为 协会 会 员 ， 和 否则 将 无 法 访问 该 页 面 ， 如 图 18-6 
所 示 。 

显示 会 员 详细 信息 页 面 的 实现 文件 是 show.php。 
协会 会 员 在 成 功 登 录 以 后 , 在 浏览 其 他 用 户 的 概要 信 


息 时 , 单 击 某 一 会 员 即 可 打开 显示 会 员 详细 信息 的 页 一 -一 各 
面 。 在 这 一 过 程 中 ， 概 要 信息 页 面 需要 提供 当前 用 户 章 
的 自动 编号 ID， 并 附加 到 URL 中 ， 显示 会 员 详细 信 图 18-6 会 员 专属 页 面 会 
息 页 面 showphp 通过 $_GET[] 方 式 获取 ID 的 值 ， 并 进行 相应 的 处 理 。 该 页 面 的 部 分 代码 如 下 所 示 ; 受 
<2php 有 
if($_SESSION['s name'] == "") 统 


1 
echo "对 不 起 ， 你 不 是 本 协会 的 会 员 ， 没 有 权限 执行 该 操作 ! "; 
echo "<script language=javascript>alert(" 对 不 起 ， 你 不 是 本 协会 的 会 员 ， 没 有 权限 执行 该 
操作 ! ') ;history.go(-1)</script>"; 
exit; 


$id = $ GET['id']; 
$query = "SELECT * FROM $user WHERE id = '$id'"; 
$result = mysql query ($query); 
$num = mysql num rows ($result); 
if ($num) 
{ 

$row = mysql_ fetch row($result); 

$id = $row[0]; 

$userid = $row[1]; 

$check = $row[2]; 

Sname = $row[3]; 

$pass = $row[4]; 

$sex = $row[5]; 

Semail = $row[6]; 

S$oicqg = $row[7]; 

$userfrom = $row[8]; 

Scity = $row[9]; 

$address = $row[10]; 

S$zip Srow[11]7 

Stel = $row[12]; 

$edu = $row[13]; 

Shangye = $row[14]; 


UI 


上 


PHP+MySQL 候 钊 省 


{ 

echo "对 不 起 ， 没 有 找到 相应 的 记录 !"; 

} 
2 5 
把 显示 模块 的 相关 代码 文件 存放 在 Apache 目录 下 的 htdocs\hy 子 目录 下 , 打开 正 浏览 器 , 在 地 区 
址 栏 中 输入 http://localhost/hy/alluser.php， 按 Enter 键 会 显示 类 似 于 如 图 18-5 所 示 的 页 面 。 然 后 ， 假 
设 单 击 会 员 “ 宋 岩 岩 ”信息 的 某 个 超级 链接 ， 会 显示 如 图 18-7 所 示 的 页 面 。 
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图 18-7 显示 会 员 详 细 信 息 


这 里 需要 注意 的 是 ， 由 于 该 页 面 只 有 协会 登录 会 员 才 能 访问 ， 所 以 在 执行 上 述 操作 前 ， 用 户 必 
须 已 经 成 功 登 录 。 


18.6 ”会员 中 心 模块 


会 员 中 心 模块 主要 涉及 注册 会 员 的 登录 及 其 成 功 登录 后 所 能 执行 的 一 些 操作 ， 这 些 操作 包括 会 
员 的 登录 与 注销 、 会 员 个 人 信息 的 修改 、 退 出 协会 及 功能 扩展 的 用 户 中 心 。 


18.6.1 用 户 登 录 与 注销 


通常 情况 下 ， 当 用 户 成 为 网 站 的 会 员 后 都 将 享有 一 些 特殊 的 功能 ， 如 发 表 日 志 、 文 章 、 上 传 
图 片 等 。 然 而 ， 这 些 操作 只 有 用 户 成 功 登 录 以 后 才能 执行 ， 从 而 体现 出 访问 者 和 正式 会 员 之 间 的 
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差别 ， 以 使 更 多 的 用 户 加 入 。 本 案例 将 介绍 协会 
会 员 如 何 进行 登录 以 及 登录 之 后 的 页 面 显 示 和 
相关 操作 。 

1. 用 户 登 录 

用 户 登 录 页 面 的 实现 文件 是 sidebarphp, 它 
通过 判断 用 户 是 否 登 录 来 显示 不 同 的 页 面 。 如 果 
用 户 未 登录 ， 将 显示 类 似 于 如 图 18-8 所 示 的 用 
户 登 录 页 面 ; 如 果 用 户 已 经 登录 成 功 , 将 显示 类 
似 于 如 图 18-9 所 示 的 用 户 信息 页 面 。 

该 页 面 中 实现 上 述 功能 的 代码 如 下 所 示 : 


18-8 用 户 登录 


图 18-9 用 户 成 功 登录 


<?php echo $ SESSION['s id'];?> 

<label for="inputtext1"> 会 员 ID: </label> 

<?php echo $ SESSION['s userid'];?> 

<label for="inputtext2"> 你 可 进行 的 操作 :</label> 

<input id="inputtext2" type="submit" name="inputtext2" value= 
"注销 ”/> 

<p><a href="usercenter.php"> 会 员 中心 </a></p> 

<p><a href="mdyuser.php?id=<?php echo $ SESSION['s id'];?>"> 


修改 个 人 信息 </a></p> 
<p><a href="killme.php?id=<?php echo $_SESSION['s_id'];?>"> 我 
要 退出 协会 </a></p> 
<?php 
if($_SESSION['a name'] != "") 第 
{ te 
?> 2 
<p><a href="admin/index.php"> 管 理 员 中 心 </a></p> 会 
<?php } ?> 员 
</fieldset> 管 
</form> 
<?php 统 
和 
</div> 
</div> 


在 sidebarphp 页 面 中 ， 会 员 输 入 自己 的 会 员 ID 和 密码 后 ， 单 击 【 登 录 】 按 钮 将 提交 到 
loginchk.php 页 面 进行 处 理 。 该 页 面 的 具体 代码 如 下 所 示 : 


<?php 

session start(); 

session register("s 1d")? 

Session register("s userid"); 

session register('s name'); 

require "conn.php"; 

require "top.php"; 

2 

<meta http-equiv="Content-Type" content="text/html; charset=gb2312" /> 
<title><?php echo $myschool;?>_ <?php echo $orgname;?> 会 员 登 录 验 证 </title> 
<link href="default.css" rel="stylesheet" type="text/css" /> 

<?php 

// 获 取 登 录 页 面 传递 来 的 用 户 信息 

Stuserid = $ POsT['userid']; 

Stpass = $ POST['pass']; 

$query = "SELECT * FROM user WHERE userid = '$tuserid'"; 

// 处 理 中 文 乱码 

$result = mysql query ($query); 

Snum = mysql num rows ($result); 


// 判 断 是 否 存在 当前 会 员 
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在 页 面 loginchk.php 中 首先 包含 进 相应 的 文件 以 备 后面 使 用 ， 其 次 是 获取 登录 页 面 传递 过 来 的 
参数 ， 接 着 查询 相应 的 数据 库 表 以 验证 当前 登录 用 户 的 信息 ; 最 后 根据 验证 结果 执行 相应 的 操作 ， 
如 果 用 户 提交 的 会 员 ID 和 密码 正确 ， 那 么 将 用 户 的 一 些 信息 保存 到 Session 中 ， 并 提示 会 员 登 录 成 
功 ; 否则 返回 给 登录 用 户 相 应 的 出 错 提示 信息 。 

2. 用 户 注销 

当 登 录用 户 完成 操作 时 ， 通 常情 况 下 需要 单 击 类 似 于 【注销 】 的 按钮 以 破坏 登录 时 创建 的 
Session， 从 而 有 效 地 保证 自己 信息 的 安全 。 这 里 用 户 注销 的 实现 文件 是 logoutphp， 它 的 具体 代码 
如 下 所 示 : 
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= 


18.6.2 ”会员 个 人 信息 修改 


由 于 会 员 信息 中 有 些 可 能 会 发 生变 动 ， 所 以 通常 情况 下 ， 网 站 都 会 为 会 员 提供 个 人 信息 修改 的 
功能 。 本 案例 也 同样 如 此 ， 在 会 员 登 录 成 功 以 后 单 击 【 修 改 个 人 信息 】 超 级 链接 就 能 转向 会 员 个 人 
信息 修改 页 面 mdyuser.php。 该 页 面 显示 与 注册 时 类 似 , 只 不 过 这 里 各 项 已 经 被 赋予 了 与 该 会 员 相 对 
应 的 值 ， 它 的 部 分 代码 如 下 所 示 : 


通常 情况 下 ， 在 会 员 修改 自己 的 信息 时 ， 会 员 ID 是 不 能 修改 的 ， 它 作为 会 员 的 标识 一 直到 会 


员 离 开 协 会 。 用 户 在 修改 完 个 人 信息 后 ， 单 击 【 确 定 】 按 钮 将 提交 给 mdyuserok.php 页 面 进行 处 理 。 
该 页 面 的 部 分 代码 如 下 所 示 : 
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18.6.3 ”退出 协会 及 扩展 功能 


用 户 加 入 协会 要 经 过 管理 员 的 批准 ， 但 用 户 如 果 想 退出 协会 就 会 很 简单 ， 只 要 成 功 登录 后 单 击 
【我 要 退出 协会 】 超 级 链接 提交 给 killme php 页 面 进行 处 理 , 该 页 面 将 执行 删除 用 户 在 用 户 信息 表 中 
的 记录 ， 并 返回 执行 结果 。 它 的 具体 代码 如 下 所 示 : 


<link href="default.css" rel="stylesheet" type="text/css" /> 


<?php 
// 获 取 传 递 来 的 用 户 信息 
Stid = $ GET['id']7 


$query = "DELETE FROM user WHERE id = "Stid'"7 
// 处 理 中 文 乱码 

//mysql query("set names gb2312") 7 
$result = mysql query(Squery) 7 
Snum = mysql affected rows () 7 

// 判 断 是 否 删除 成 功 

if($num>0) 

{ 


[| 


册 


echo "会 员 删 除 成 功 ! <hr><p>"; 

echo "登录 会 员 是 : ".$_SESSION['s_name']."<p>"; 

echo "现在 你 已 经 退出 [" .$orgname ."] ,将 不 再 享有 本 协会 提供 的 服务 。"; 
echo "<br> 返 回 <a href='index.php'> 首 页 </a>"; 

SSESSTOMN[YS GE wey 

$_SESSION['s userid'] = ""7 

$ SESSION['s name'] = ""7 
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else 
{ 
echo "会 员 删除 失败 ! <hr><p>"; 
echo "当前 会 员 不 存在 ! "; 
echo "<br><a href='reg.php'> 请 先 加 入 协会 ， 再 进行 操作 ! </a>"; 
} 
?> 
<?php 
require "foot.inc"; 
?> 


当 几 记过 出 娘 会 后， 用 户 的 oo 
记录 将 从 用 户 信 息 表 中 删除 ， 用 5 必 ww oesssnroreccnn mw 
户 将 不 再 具有 登录 本 系统 的 权 
限 , 除非 用 户 再 次 进行 注册 成 功 。 

关于 用 户 中 心 ， 这 里 只 是 提 
供 一 个 扩展 功能 的 框架 ， 用 于 说 
明 在 通常 情况 下 会 员 所 具有 的 一 
些 功 能 ， 具 体 如 图 18-10 所 示 。 
读者 可 以 通过 实现 这 些 功 能 来 i 
掌握 PHP 编程 的 相关 知识 及 其 — 
构思 。 图 18-10 会 员 权 限 


强人 网 络 协会 会 员 的 权限 : 


445 


18.7 ”管理 模块 


管理 模块 用 于 实现 管理 员 对 会 员 信息 进行 管理 。 在 进入 后 台 管 理 之 前 , 管理 员 必须 先进 行 登录 ， 
只 有 成 功 登 录 的 管理 员 才能 对 会 员 信息 进行 管理 ， 关 于 管理 员 登 录 的 实现 与 普通 会 员 的 实现 基本 相 
同 , 这 里 不 再 进行 介绍 。 除 了 对 会 员 信息 进行 管理 之 外 , 管理 员 还 具有 普通 会 员 所 具有 的 全 部 功能 ， 
比如 修改 个 人 信息 、 注 销 等 。 该 页 面 如 图 18-11 所 示 。 
避 中 软 太 学 _ 台 人 网 绍 协会 _ 后台 于 理 一 Microsoft Internet Erplorer 
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图 18-11 后 台 管理 页 面 


这 里 只 重点 介绍 如 何 顺 利 管理 会 员 的 验证 状态 、 如 何 提升 会 员 为 管理 员 、 如 何 推荐 优秀 会 员 及 
删除 现 有 会 员 等 方面 的 内 容 。 


18.7.1 ”管理 会 员 验 证 状态 


当 用 户 首次 申请 要 求 成 为 协会 会 员 时 ， 管 理 员 首 先 查 看 该 申请 会 员 的 信息 ， 然 后 对 其 信息 的 真 
实 性 及 本 人 情况 进行 调查 、 审 核 ， 最 后 根据 审核 结果 执行 相应 的 操作 。 该 操作 的 实现 文件 是 
validate.php， 它 根据 管理 员 操作 传递 过 来 的 参数 判断 是 通过 验证 ， 还 是 取消 验证 。 该 页 面 的 具体 代 
码 如 下 所 示 : 


<?php 
session start (); 
require "conn.php"; 
require "top.php"; 
if($_SESSION['a name'] == "") 
| 
echo "对 不 起 ， 你 不 是 本 协会 网 站 的 管理 员 ， 没 有 权限 执行 该 操作 !"; 
echo "<script language=javascript>alert (' 对 不 起 ， 你 不 是 本 协会 网 站 的 管理 员 ， 没 有 权限 
执行 该 操作 ! ') ;history.go(-1)</script>"; 
ei 


人 
<meta http-equiv="Content-Type™" content="text/html; charset=gb2312" /> 
<title><?php echo $myschool;?> <?php echo $orgname;?> 修改 会 员 状 态 结果 </title> 
<link href="../default.css" rel="stylesheet" type="text/css" /> 
<?php 

// 取 得 提交 过 来 的 信息 

$tid = $ GET['id']; 

Staction = $ GET['action']; 

$query = ""; 

// 判 断 操作 的 方式 

// 如 果 taction 的 值 为 tg, 则 设置 check 字段 为 1 

if($taction == "tg") 

{ 


瑟 小 


山 


$query = "UPDATE Suser SET checkflag "1' WHERE id = '$tid'"; 


} 


elseif ($taction == "qx") 


{ 


'0' WHERE id 


$query = "UPDATE Suser SET checkflag "tid"™s 


尝 洲 腊 式 i 冰 


} 
// 执 行 更 新 操作 
$result = mysql query ($query) 
or die ("查询 出 现 错误 : " . mysql error()); 
$rowid = mysql affected rows(); 
if ($rowid>0) 
{ 
echo "修改 会 员 状 态 成 功 ! <hr><p>"; 
echo "修改 状态 的 协会 会 员 是 : ".$_SsESSION['s_name']."<p>"; 
echo "<br><a href='index.php'> 修 改 其 他 </a>"; 
} 
else 
{ 
echo "修改 会 员 状 态 失败 ! <hr><p>"; 
echo "会 员 : [".$_SESSION['s_name']."]"; 
echo "状态 修改 失败 ! "; 
echo "<br><a href='index.php'> 重 新 进行 修改 </a>"; 


mysql close(); 
?> 
</div> 
<?php 
require "foot.inc"; 


让 


于 该 页 面 的 执行 操作 对 用 户 的 影响 很 大 ， 所 以 只 有 成 功 登录 的 管理 员 才 能 有 权限 进入 ， 并 执 


行 会 员 状 态 的 修改 操作 。 
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18.7.2 ”提升 会 员 为 管理 员 


管理 员 的 变动 也 是 经 常 出 现 的 ， 所 以 管理 员 有 权限 将 某 一 会 员 提升 为 管理 员 ， 该 会 员 被 提升 为 
管理 员 后 将 具有 管理 员 的 所 有 功能 。 实 现 提升 会 员 为 管理 员 的 文件 是 updatephp， 它 的 具体 代码 如 
下 所 示 : 


18.7.3 ”删除 会 员 


当 协 会 的 会 员 已 不 再 符合 协会 对 会 员 的 要 求 ， 或 者 会 员 退 出 了 协会 而 并 没有 在 会 员 管理 系统 上 
单 击 【 我 要 退出 协会 】 超 级 链接 时 ， 都 需要 管理 员 进行 删除 操作 来 删除 会 员 信息 ， 以 保证 系统 数据 
的 正确 性 。 删 除 会 员 的 实现 文件 是 deluser.php， 它 的 具体 代码 如 下 所 示 : 


册 矶 由 


30N> 


hr hs 
胸 融 > 


N 


7 


Stuserid = $ GET['userid']7 
$auserid = $ SESSION['"a name']; 


// 判 断 删除 的 会 员 是 否 是 当前 管理 员 自 己 的 信息 
if($auserid == Stuserid) 
{ 
echo "会 员 删 除 失败 ! <hr><p>"; 
echo "当前 管理 员 不 能 删除 自己 的 信息 ! "; 
echo "<br><a href='index.php'> 进 行 其 他 操作 ! </a>"; 
P 


| 
echo "会 员 删 除 成 功 ! <hr><p>"; 
echo "删除 的 会 员 是 : " .Stname ."<p>"7 
echo "现在 该 会 员 已 经 退出 [" .Sorgname ."] ,将 不 再 享有 本 协会 提供 的 服务 。"; 
echo "<br> 返 回 <a href='index.php'> 进 行 其 他 操作 </a>"; 


else 
{ 
对 $query = "DELETE FROM user WHERE id = '$tid'"; 
卫 // 处 理 中 文 乱码 
F4 //mysql query("set names gb2312"); 
六 $result = mysql query(Squery) 7 
思 Snum = mysql affected rows () 7 
完 // 判 断 是 否 删除 成 功 
3 if ($num>0) 
学 
习 
手 
册 


else 
{ 
echo "会 员 删 除 失败 ! <hr><p>"; 
echo "当前 会 员 不 存在 ! "; 
echo "<br><a href='index.php'> 进 行 其 他 操作 ! </a>"; 
} 
} 
?> 
<?php 
require "foot.inc"; 
?> 


在 删除 会 员 页 面 deluserphp 中 ， 首 先 判断 当前 用 户 是 否 是 管理 员 ， 如 果 是 则 继续 执行 后 面 的 操 
作 ， 否 则 停止 执行 后 面 的 脚本 程序 ， 并 返回 相应 的 出 错 提 示 信息 ， 其 次 判断 所 删除 的 会 员 是 否 是 管 
理 员 自 己 ， 因 为 管理 员 是 能 从 会 员 中 产生 的 ， 所 以 每 个 管理 员 信息 表 中 的 记录 都 可 以 从 用 户 信息 表 
中 找到 相应 的 会 员 信息 , 之 所 以 如 此 , 当前 管理 员 不 能 删除 自己 的 信息 。 接着 获取 要 删除 用 户 的 D， 
并 执行 相应 的 删除 操作 ; 最 后 将 删除 会 员 的 操作 结果 返回 给 当前 管理 员 。 


18.8 ”系统 首页 实现 


该 会 员 管理 系统 的 首页 共 分 为 4 个 部 分 ， 之 所 以 这 么 做 是 为 了 实现 模块 化 设计 ， 因 为 在 一 个 网 


站 中 有 一 些 内 容 是 许多 页 面 都 要 用 到 的 ， 每 次 用 到 都 要 设计 一 次 ， 既 费事 又 费力 ， 也 不 能 保证 两 次 
做 的 工作 完全 一 样 。 为 了 解决 这 个 问题 ， 通 常 采用 的 方法 就 是 把 通用 的 部 分 提取 出 来 单独 保存 在 一 
个 文件 中 ， 然 后 在 需要 的 地 方 包含 进来 就 行 了 。 下 面 就 来 对 首页 的 这 4 个 部 分 进行 介绍 。 


18.8.1 ”顶部 模块 gE 


顶部 模块 的 实现 文件 为 top.php， 它 主要 包括 本 会 员 管 理 系统 的 导航 菜单 、 网 站 名 称 、 子 名 称 及 
网 站 的 宣传 图 片 等 方面 的 内 容 , 它 在 用 户 浏览 index.php 页 面 时 自动 加 载 进来 。 该 部 分 的 显示 效果 如 
图 18-12 所 示 。 


互 浊 


册 


中 软 大 学 
强人 网 络 协会 会 员 管理 闲 尖 
首页 。 加 入 协会 。 所 有 会 员 ”查找 会 员 ”用 户 中 心 。 介 理 登录 


。 品 联系 我 们 ” 白 本 站 地 图 


全 
b=3 
员 
管 
理 
系 
统 


图 18-12 首页 项 部 的 显示 效果 


从 图 18-12 所 示 的 顶部 中 可 以 看 出 ， 它 分 别 链 接 了 【加 入 协会 【所 有 会 员 】【 查 找 会 员 】【 用 
户 中 心 】 和 【管理 登录 】 等 子 页 面 的 超级 链接 ， 这 些 都 是 会 员 类 网 站 上 经 常见 到 的 部 分 ， 它 们 分 别 
对 应 于 前 面 介绍 的 注册 模块 、 显 示 模 块 、 查 询 模块 、 会 员 中 心 模块 和 管理 模块 。 通 常情 况 下 ， 人 们 
称 这 些 超级 链接 为 导航 菜单 。 这 些 是 在 top.php 文件 中 手动 添加 的 ， 在 实际 应 用 中 ， 可 以 专门 为 这 
些 导航 菜单 在 数据 库 中 创建 一 个 表 ， 通 过 后 台 进 行 添加 和 修改 导航 菜单 ， 在 前 台 通过 查询 数据 库 调 
用 即 可 ， 这 样 就 可 以 实时 、 动 态 地 控制 这 些 导航 菜 单 了 。 实 现 顶 部 代码 如 下 所 示 : 


<div id="header"> 
<div id="topmenu"> 
<ul> 
<li><a href="index.php"” id="topmenul" accesskey="1"” title=""> 首 页 
</a></1i> 
<li><a href="http://www.itying.net/" id="topmenu2" accesskey="2" 
title=" "> 联系 我 们 </a></1i> 
<li><a href="http://www.itying.net/" id="topmenu3" accesskey="3" 
title=""> 本 站 地 图 </a></1i> 
</ul> 
</div> 
<div id="logo"> 
<hl><a href="<?php echo S$orgweb;?>"><?php echo $myschool;?></a></h1l> 
<h2><a href="http://www.itying.net/"><?php echo $orgname;?> 会 员 管理 系统 
</a></h2> 
</div> 
</div> 
<div id="menu"> 
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上 述 代码 看 起 来 很 简单 ， 也 没有 使 用 表格 来 控制 信息 显示 的 位 置 ， 那 么 上 面 的 显示 效果 是 如 何 
实现 的 呢 ? 其 实在 本 案例 中 ,使 用 了 CSS〈 层 莹 样式 表 ) 来 控制 信息 的 显示 位 置 及 效果 ， 每 个 页 面 通 
过 包含 default.css 文件 来 使 用 定义 的 样式 。 


18.8.2 ” 右 部 模块 


右 部 模块 的 实现 文件 为 sidebarjsp， 它 主要 用 于 显示 用 户 的 登录 情况 、 最 新 加 入 及 本 站 统计 方 
面 的 内 容 , 它 也 是 在 用 户 浏览 index.php 页 面 时 自动 加 载 进来 的 。 其 中 ,关于 用 户 登 录 方 面 的 内 容 在 
会 员 中 心 模块 中 已 经 介绍 ， 在 此 不 再 提 及 ， 只 介绍 关于 最 新 加 入 和 本 站 统计 方面 的 内 容 ， 有 具体 的 实 
现代 码 如 下 所 示 : 


在 上 述 代码 所 示 的 页 面 中 ， 只 上 下 滚动 显示 最 新 加 入 的 5 位 会 员 ， 当 将 鼠标 移动 过 去 时 滚动 停 
止 , 用 户 可 以 单 击 相应 的 会 员 姓名 来 查看 该 会 员 的 详细 信息 。 当 然 , 前 提 是 当前 用 户 已 经 登录 成 功 。 


18.8.3 ”主体 和 底部 模块 


主体 模块 紧 挨 着 首页 左边 登录 模块 的 右边 部 分 ， 它 的 实现 文件 为 bestframe.php， 主 要 用 于 展示 
协会 推荐 的 优秀 会 员 ， 如 果 该 类 会 员 多 于 两 个 那么 将 分 多 页 进行 显示 ， 并 通过 判断 best 字段 的 值 是 
否 为 1， 来 决定 是 否 显示 该 条 记录 的 会 员 人 信息。 实现 该 页 面 的 部 分 代码 如 下 所 示 : 


PHP+MySQL 候 镍 洲 外 中 这 


<1i> 会 员 编号 : <a href="show.php?id=<?php echo $id;?>"><?php echo $id;?></a></1i> 
<1i> 会 员 名 称 : <a href="show.php?id=<?php echo $id;?>"><?php echo $userid;?> 


</a></1i> 

<1i> 真 实 姓名 : <a href="show.php?id=<?php echo $id;?>"><?php echo $name;?> 
</a></1i> 

<1i> 会 员 姓名 : <a href="show.php?id=<?php echo $id;?>"><?php echo $sex;?></a></1i> 
<?php 

if($best == "1") 


wn 
s 
部 
a 
mY 
1 


mw 人 
片 
站 
人 
[| 


$best = 章 

全 

<1i> 优 秀 会 员 ，<a href="show.php?id=<?php echo $id;?>"><?php echo $best;?> 丛 

</a></1i> 理 

系 

</td> 统 
</tr> 


<tr width="200"> 
<td colspan="2" width="200"><div id="welcome"> 
<strong> 会 员 简 介 : </strong></div><?php echo $intro;?></td> 
</tr> 
</table> 
</td> 
</tr> 


<?php 
} 
} 
LS 
| 
echo "没有 记录 "; 
h 
?> 
</table> 


底部 模块 的 实现 文件 为 EEC 
footinc， 它 主要 用 于 包括 版 权 信 | 些 避 是 ww poeador Po/ nen pe 
息 、 底 部 导航 菜单 及 其 他 的 一 些 相 


关 信 息 ， 如 图 18-13 所 示 。 


在 这 里 需要 说 明 的 是 , 本 会 员 GET 
管理 系统 中 的 部 分 内 容 并 没有 实 图 18-13 ”底部 显示 效果 
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现 ， 因 为 本 案例 主要 介绍 会 员 的 注册 、 登 录 及 信息 修改 ; 会 员 信息 的 查询 、 显 示 及 会 员 的 管理 等 方 
面 的 内 容 ， 有 兴趣 的 读者 可 以 在 本 系统 的 基础 上 加 以 扩展 ， 以 使 该 系统 更 加 完善 。 


18.8.4 其 他 通用 文件 


在 本 案例 中 ， 基 本 上 每 个 页 面 都 包含 了 这 两 个 文件 : connphp 和 default.css。 其 中 ，conn.php 
文件 用 于 存放 数据 库 及 表 名 称 的 变量 及 值 、 创 建 到 MySQL 数据 库 的 连接 及 其 他 一 些 系统 中 经 常用 
到 的 变量 及 其 相对 应 的 值 。 它 的 具体 代码 如 下 所 示 : 


default.css 文件 用 于 控制 相关 数据 的 显示 方式 及 位 置 。 下 面 对 其 中 的 一 些 代 码 进行 说 明 : 


上 述 代 码 应 用 于 整个 页 面 ，background 用 于 设置 对 象 的 背景 样式 ，font 用 于 设置 对 象 的 文本 格 
式 ; color 用 于 设置 对 象 的 文本 颜色 ， 无 默认 值 。 


上 述 代码 作用 于 <div id="header"> 和 </div> 之 间 的 对 象 , width 用 于 设置 对 象 的 宽度 ; height 用 于 
设置 对 象 的 高 度 ，margin 用 于 设置 对 象 四 边 的 外 补丁 ， 默 认 值 为 0; 


上 述 代码 作用 于 <div id="topmenu"> 和 </div> 之 间 的 对 象 ，display 用 于 设置 或 检索 对 象 是 否 进行 
显示 及 如 何 显示 ; float 用 于 指出 对 象 是 否 及 如 何 浮动 ，padding 用 于 检索 或 设置 对 象 四 边 的 内 补丁 ， 
对 于 td 和 也 对 象 而 言 默 认 值 为 1， 其 他 对 象 的 默认 值 为 0; text_transform 用 于 检索 或 设置 对 象 中 的 
文本 大 小 写 ，text-decoration 用 于 检索 或 设置 对 象 中 的 文本 装饰 ，font-size 用 于 设置 或 检索 对 象 中 的 
字体 尺寸 ，font-weight 用 于 设置 或 检索 对 象 中 的 文本 字体 的 粗细 。 


第 19 章 投票 管理 系统 


/ 学 习 目 标 |ouiecuve 


开发 一 个 投票 网 站 或 者 在 网 站 中 加 入 投票 模块 ， 能 够 很 好 地 获取 反馈 信息 ， 使 人 们 能 够 快速 并 
准确 地 进行 信息 统计 。 使 用 PHP 技术 开发 的 在 线 投票 管理 系统 ， 具 有 代码 重用 、 安 全 、 容 易学 习 等 
多 种 特点 。 

PHP 在 线 投票 管理 系统 具有 投票 、 投 票 统计 、 投 票 管理 、 选 项 管理 等 功能 ， 在 系统 的 开发 过 程 
中 ， 涉 及 到 正则 表达 式 、PEAR 包 等 多 种 PHP 高 级 技术 。 


FY 内 容 摘要 |avstract 


> 了 解 在 线 投票 系统 的 开发 流程 

> 掌握 系统 的 各 个 模块 

> 掌握 PHP 代码 的 编写 和 调试 

> 掌握 在 系统 中 使 用 PEAR 包 

> 掌握 在 系统 中 使 用 正则 表达 式 校 验 

> 掌握 使 用 PHP 中 的 公共 代码 

> 熟练 掌握 并 解决 PHP 连接 数据 库 问 题 
> 熟练 掌握 并 解决 多 种 代码 的 编写 和 调试 
> 掌握 复杂 的 SQL 语句 的 编写 和 调试 


19.1 系统 概述 


一 个 系统 从 无 到 有 ， 从 开始 开发 到 完成 ， 需 要 分 析 许 多 问题 ， 遵 循 许多 步骤 和 原则 ， 创 建 多 个 
文档 ， 以 确保 系统 进度 的 可 控 性 和 质量 的 预 估 性 。 软 件 开发 经 历 的 过 程 有 需求 分 析 、 概 要 设计 、 详 
细 设 计 等 。 每 个 过 程 完成 不 同 的 任务 ， 每 一 个 过 程 都 是 后 一 个 过 程 的 基础 ， 每 个 过 程 都 要 创建 相应 
的 文档 作为 基础 。 本 章 将 以 投票 管理 系统 为 例 ， 体 现 软件 开发 的 每 个 过 程 。 

在 网 络 上 投票 已 经 屡见不鲜 了 ， 比 如 评选 世界 十 大 奇迹 、 最 佳 主持 人 等 。 这 些 投 票 大 多 数 都 是 
针对 某 一 个 特定 对 象 而 言 的 ， 如 有 具有 固定 的 选项 ， 只 能 对 特定 的 项 目 投票 等 。 为 了 解决 上 面 这 些 问 
题 来 编写 PHP 在 线 Web 投票 系统 。 

开发 投票 管理 系统 所 需要 的 开发 工具 、 运 行 环境 、 数 据 库 分 别 为 纯 文本 编辑 器 记事 本 、PHP 
5.0+Apache 2.2、MySQL。 在 该 软件 的 开发 过 程 中 ， 使 用 PEAR 包 解 决 问题 。 该 系统 的 开发 目的 有 
两 个 ,一 个 是 供 广大 读者 学 习 ， 读 者 可 以 从 本 案例 中 学 习 到 PHP 代码 的 编写 、 运 行 和 调试 等 。 一 
是 供 公 司 使 用 , 可 以 在 本 系统 的 基础 上 加 以 改造 , 成 为 一 个 商业 性 质 的 网 站 。 本 系统 创建 完成 之 后 ， 


要 具备 下 面 的 功能 : 

。 建立 一 个 拥有 良好 交互 性 、 操 作 简 单 易 用 的 在 线 Web 网 站 。 

。 网 站 运行 要 高 效 ， 费 用 尽量 低 ， 注 重 实用 性 。 

。 网 站 能 够 自动 执行 添加 投票 项 目 和 项 目 选项 。 

。 对 于 投票 数据 ， 能 够 及 时 分 析 。 

。 对 于 投票 项 目 和 相应 选项 ， 能 够 进行 继续 调整 。 

PHP 在 线 Web 投票 系统 具有 投票 、 投 票选 项 管理 、 投 票 统计 、 投 票 管理 4 个 功能 。 投 票 功 能 可 
以 使 客户 自由 地 投 出 一 票 。 投 票选 项 管理 对 参加 投票 的 选项 进行 管理 ， 如 添加 、 删 除 、 修 改 等 。 投 
票 统计 对 投票 信息 进行 统计 。 投 票 管理 具有 添加 投票 项 目 、 添 加 投票 选项 、 修 改 、 删 除 、 显 示 等 功 
能 。 其 关系 如 图 19-1 所 示 。 


投票 管理 系统 


a i 


图 19-1 投票 管理 系统 模块 


19.2 ”数据 库 实现 


在 实现 整个 系统 之 前 ， 首 先 需要 对 实现 功能 进行 初步 的 讨论 并 分 析 结 构图 ， 这 些 工 作 在 上 一 个 
小 节 中 已 经 完成 。 在 分 析 结构 之 后 ， 进 行 编码 设计 之 前 ， 还 要 考虑 系统 的 实现 需要 哪些 数据 表 ， 数 
据 表 中 包括 哪些 字段 ， 这 些 字段 用 来 做 什么 等 ， 下 面 将 对 系统 中 使 用 到 的 数据 表 及 字段 进行 详细 
说 明 。 

PHP 在 线 投票 管理 系统 具有 三 个 表 ， 分 别 为 投票 项 目 表 mu、 投 票 项 目 选项 表 xuan、 投 票 信息 
表 xin。 这 些 表 的 字段 信息 如 下 所 示 。 

1. 投票 项 目 表 mu 

mnu 表 主要 用 来 保存 投票 的 项 目 名 称 ， 如 世界 十 大 奇迹 、 最 佳 新 闻 等 。 该 表 字 段 信息 如 表 19-1 
所 示 。 


首页 


表 19-1 投票 项 目 表 字 段 信息 


Tinyint(3) 
varchar(30) 


varchar(15) 


投票 项 目 序列 号 
投票 项 目 名 称 
投票 项 目 投票 方式 


2. 投票 项 目 选项 表 xuan 
该 表 主 要 用 来 保存 每 个 项 目 选 项 ， 如 投票 项 目 世界 十 大 奇迹 中 的 长 城 、 金 字 塔 等 。 该 表 字 段 信 
息 如 表 19-2 所 示 。 


忌 站 


册 


投 
管 
理 
系 
统 


表 19-2 ”投票 项 目 选项 表 字 段 信息 


字 段 数据 类 型 允许 空 备注 
xuan id Tinyint(3) 否 投票 选项 序列 号 
mu id | varchar(25) 否 投票 项 目 序列 号 
xuan name varchar(50) 否 投票 选项 名 称 


3. 投票 信息 表 xin 
该 表 主 要 用 来 保存 参加 投票 的 个 人 信息 和 投票 项 目 。 投 票 信息 表 主 要 用 来 解决 投票 作弊 问题 ， 
一 个 浏览 者 只 能 投 一 次 票 。 该 表 字段 信息 如 表 19-3 所 示 。 
表 19-3 ”投票 信息 表 字 段 信息 

字段 | 数据 类 型 | 允许 空 | 人 
xin id | Te | 否 投票 序列 号 
mu id | vaG) | 否 | 投票 项 目 序列 号 
xuan_name | vahals) | 下 | 投票 的 选项 名 称 
xin_ time | vachao | | 投票 时 间 
xin name | va | 百 投票 人 名 称 
xin_ phone | va | 百 | 投票 人 联系 方式 
xin_zhen | va | | 投票 人 身份 证 号 
xin_piao [am | | 投票 数 


19.3 首页 


在 编写 软件 代码 之 前 ， 需 要 创建 一 个 文件 夹 ， 用 来 保存 编写 的 PHP 页 面 文 件 。 在 
C:\Web\WApache\htdocs 文件 夹 下 ， 建 立 一 个 文件 夹 tou 作为 保存 文件 的 地 方 ， 并 且 在 tou 文件 夹 下 建 
立 images 文件 夹 ， 作 为 保存 该 软件 所 需 的 图 片 。 


19.3.1 实现 公共 代码 
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为 了 实现 该 软件 的 代码 重用 ， 逻 辑 与 表示 相互 分 离 ， 这 里 需要 编写 一 个 PHP 页 面 ， 作 为 通用 模 
块 ， 实 现 PHP 对 MySQL 数据 库 连 接 及 SQL 语句 执行 。 打 开 记 事 本 ， 输 入 下 列 代码 : 


<?php 
class Pconf{ 
public function getC()1{ 
$link=mysql connect ("localhost","root","root"); 
} 

public function ins($sql){ 
$link=mysql_ connect ("localhost", "root", "root"); 
mysql select db("piao"); 
$result=mysql query($sql); 


mysql close(); 


上 
2 
将 上 述 代码 保存 ， 文 件 名 为 piaoCon.php。 在 上 述 代码 中 创建 了 一 个 类 Pcon， 该 类 具有 两 个 方 
法 ， 一 个 方法 是 getCO， 用 来 创建 一 个 到 MySQL 数据 库 的 连接 。 另 外 一 个 方法 是 inSO， 用 来 执行 
一 个 更 新 语句 ， 如 添加 、 删 除 、 修 改 等 。 


19.3.2 ”实现 首页 


滤 


首页 是 一 个 网 站 的 门面 ， 好 的 首页 会 带 来 更 高 的 点 击 率 。PHP 在 线 投票 系统 首页 ， 具 有 连接 不 
同 的 模块 ， 并 显示 当前 最 新 投票 项 目的 作用 。 打 开 记事 本 ， 实 现 首页 的 编写 ， 输 入 下 列 代码 : 


<HEAD> 
<TITLE>IT 在 中 国 http://www.itzcn.com</TITLE> 
<META HTTP-EQUIV="Content-Type" CONTENT="text/html; charset=GB2312"> 
<link rel="stylesheet" href="styles.css" type="text/css"> 
</HEAD> 
<BODY BGCOLOR=#FFFFFF LEFTMARGIN=0 TOPMARGIN=0 MARGINWIDTH=0 MARGINHEIGHT=0> 
<table width="100%" border="0" cellspacing="0" cellpadding="0"> 
<Er> 
<td height="137" valign="top" background="images/bgt.gif"> 
<table width="100" border="0" cellspacing="0" cellpadding="0"> 
<tr> 
<td><img src="images/head.jpg" width=378 height=96 alt=""></td> 
<td><img src="images/thead.gif" width=372 height=31 alt=""><br> 
<img src="images/name.jpg" width=372 height=27 alt=""><br> 
<a href="http://www.itzcn.com"><img src="images/slogan.jpg" width=372 
height=18 border=0 alt=""><br> 
</a><img src="images/index 05.jpg" width=372 height=20 alt=""></td> 
</tr> 
</table> 
<table width="750" border="0" cellspacing="0" cellpadding="0"> 
LE 
<td> 
<ahref=index.php><img src="images/ml.gif" width=56 height=29 alt=""></a> 
<a href="toul.php"><img src="images/m2.gif"width=73 height=29 alt=""></a> 
<a href="tou.php"><img src="images/m4.gif" width=84 height=29 
alt=""></a> 
<a href='muguanl.php'><img src="images/m5-.gif" width=78 height=29 


本 
< 
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alt=""></a> 

<a href="muadd0 .php"><img src="images/m3.gif" width=68 height=29 
alt=""></a> 

<a href="Calendar.php"><img src="images/m6.gif" width=68 height=29 
alt=""></a> 
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将 上 述 代码 保存 ， 文 件 名 为 index.php， 将 其 保存 在 tou 文件 夹 下 。 在 上 述 代 码 中 ，HTML 标记 
用 来 规定 首页 显示 的 样式 , 在 HTML 标记 中 , 嵌入 了 两 段 PHP 程序 段 , 这 两 段 PHP 程序 使 用 require 
语句 分 别 引入 了 testphp 文件 和 testl.php 文件 。 这 两 个 文件 主要 用 来 显示 MySQL 数据 库 的 数据 。 

1. 实现 test.php 页 面 

test.php 页 面 主要 显示 当前 最 新 的 投票 项 目 , 通过 该 页 面 可 以 直接 参加 投票 。 为 了 实现 该 页 面 功 
能 ， 打 开 记事 本 ， 输 入 下 列 代码 : 


echo "可 能 获取 " . $name ." 奖 的 是 "; 
echo "<input type=hidden name=name5 value=".$id.">"; 
echo "<input type=hidden name=name6 Value=" .Sname-">"7 
$sql="select * from xuan Where mu id=$id"; 
$result=mysql query($sql); 
if ($shi==' 单 选 ') { 
while($rs=mysql fetch object ($result)) 
{ 

$id=$rs->xuan id; 

$name=$rs->xuan name; 

echo "<br><input type=radio name=name7 value=".$name.">".$name."<br>"; 


} 


mysql close(); 第 

L 19 

if ($shi-" 复 选 ') { 章 
while($rs=mysql fetch object ($result)) 

: 和 
$id=$rs->xuan id7 5 
Sname=Srs->xuan name; 忆 
echo "<br><input type=checkbox name=name7 value=".$name.">".S$name."<br>"; 

， 的 
mysql close() 7 
} 
a 
<input type=submit value= 提 交 > 
</form> 


将 上 述 代码 保存 为 testphp 文件 。 在 该 文件 中 ， 使 用 查询 语句 获取 项 目 表 中 最 新 投票 项 目 ， 并 
将 该 投票 项 目 序列 号 使 用 变量 id 保存 起 来 。 在 下 面 的 语句 中 ， 以 变量 id 作为 参数 在 表 xuan 中 进行 
查询 , 获取 最 新 项 目的 每 个 选项 , 并 根据 获得 mu_shi 字段 值 来 决定 是 单 选 投票 方式 还 是 复 选 投票 方 
式 。 一 切 判定 后 ， 以 指定 样式 在 首页 显示 最 新 投票 信息 。 

2， 实现 test1.php 页 面 

testl.php 页 面 主要 显示 投票 项 目 ， 可 以 从 中 选择 一 个 有 兴趣 的 投票 项 目 进行 投票 。 为 了 实现 该 
文件 功能 ， 打 开 记 事 本 ， 输 入 下 列 代码 : 


<p align=center><font color=pear size=5> 请 选择 要 参加 的 项 目 进行 投票 </font></p> 
<?php 
// 建立 数据 库 连 接 
$link=mysql connect ("localhost","root","root"); 
mysql select db ("piao"); 
// 获取 当前 页 数 
if( isset($ GET['page']) ){ 
$page = intval( $ GET['page'] ); 
} 
else 
{ 
Spage = 1; 


j 
// 每 页 数量 


Spage _ size = 3 
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echo "没有 记录 "; 
} 
2> 
<?php 
// 翻 页 链接 
Spage string = "'"; 
if( Spage == 1 ){ 
Spage_string .= ' 第 一 页 | 上 一 页 | '; 
4 


else{ 
Spage_string .= '<a href=?page=1> 第 一 页 </a>1<a href=?page='. (Spage-1) .'> 上 一 页 第 
</a>l'; 

Ph 

if( ($page == S$page count) || ($page count == 0) ){ 


Spage_string .= ' 下 一 页 | 尾 页 '; 

} 

elsef 
Spage_string .= '<a href=?page='. (Spage+1) . '> 下 一 页 </a>1<a href=?page=' .Spage 
count . '> 尾 页 </a>'7 

} 

echo "<br>".$page string; 

2 


将 上 述 代码 保存 为 test1l.php。 在 该 文件 中 ,主要 实现 显示 表 mu 中 的 所 有 选项 ,并 以 分 页 形式 显示 。 
打开 正 浏览 器 ， 在 地 址 栏 中 输入 http://localhost:8080/tow/index.php， 单 击 【 转 到 】 按 钮 ， 会 显 
示 如 图 19-2 所 示 的 窗口 。 
电工 


TIRETTIETIETIETCIETII 
此 站 加 ) | 简 http:yylocalhost:a0ao/towinder ptp 


瘤 洪 六 下 酒 沙 ” 山 


请 选择 要 参加 的 项 目 进行 投票 


文明 4 区 
星人 后 中 环境 
和 -页 | 上 一 页 | 下 -- 豆 尾 页 


图 版权 所 有 加 IT 在 中 国 [ww itZon. net] 
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对 于 该 软件 的 首页 实现 ， 采 用 了 逻辑 与 表示 相互 分 离 的 方法 。testphp 和 testl .php 两 个 文件 分 
别 实现 网 页 动态 数据 显示 ，index.php 文件 主要 实现 首页 静态 显示 。 


19.4 ”投票 统计 模块 


投票 统计 模块 可 以 查看 指定 投票 项 目的 投票 情况 ， 如 该 项 目 投票 参加 人 数 ， 哪 些 选 项 是 前 三 名 
等 。 投 票 统 计 模 块 是 PHP 在 线 投票 系统 分 析 模块 。 单 击 图 19-2 中 【投票 统计 】 超 级 链接 就 会 进入 
该 模块 。 进 入 模块 后 ， 可 以 选择 要 查看 的 投票 项 目的 投票 情况 ， 如 选择 中 华 环球 小 姐 评选 投票 分 析 
情况 。 


19.4.1 实现 统计 显示 页 面 


首先 实现 显示 所 有 投票 项 目的 PHP 文件 ， 该 文件 也 可 以 使 用 分 页 显示 。 打 开 记事 本 ， 输 入 下 列 
代码 : 


<HEAD> 
<link rel="stylesheet" href="styles.css" type="text/css"> 
</HEAD> 
<BODY BGCOLOR=#FFFFFF LEFTMARGIN=0 TOPMARGIN=0 MARGINWIDTH=0 MARGINHEIGHT=0> 
<table width="100%" border="0" cellspacing="0" cellpadding="0"> 
<tr> 
<td height="137" valign="top" background="images/bgt.gif"> 
<table width="100" border="0" cellspacing="0" cellpadding="0"> 
<tr> 
<td><img src="images/head.jpg" width=378 height=96 alt=""></td> 
<td><img src="images/thead.gif" width=372 height=31 alt=""><br> 
<img src="images/name.jpg" width=372 height=27 alt=""><br> 
<a href="http://www.itzcn.com"><img src="images/slogan.jpg" width=372 
height=18 border=0 alt=""><br> 
</a><img src="images/index 05.jpg" width=372 height=20 alt=""></td> 
</tr> 
</table> 
<table width="750" border="0" cellspacing="0" cellpadding="0"> 
<tr> 
<td> 
<a href=index .php><img src="images/ml.gif" width=56 height=29 alt=""></a> 
<a href="toul.php"><img src="images/m2.gif" width=73 


height=29 alt=""></a> 
<a href="tou.php"><img src="images/m4 .gif" width=84 height=29 alt=""></a> 


<a href='muguanl .php'><img src="images/m5-.gif" width=78 height=29 alt= 
"n></a> 
<a href="muadd0 .php"><img src="images/m3.gif" width=68 height=29 alt= 


mn></a> 
</td> 
</tr> 
</table> 
</td> 
过 /> 
</table> 
<table width="750" border="0" cellspacing="0" cellpadding="0" class="styles"> 
< 
<td width="81" valign="top"> 
<img src="images/rc.gif" width=41 height=70 alt=""> 
</ta> 


<td width="669"> 第 
19 
<h1> 从 当前 页 面 可 以 查看 投票 信息 </h1> 吝 
<?php 投 
// 建立 数据 库 连接 票 
$link=mysql_connect ("localhost", "root", "root"); 管 
mysql_ select db("piao"); 理 
/7 获取 当前 页 数 全 
if( isset($ GET['page']) ){ 
$page = intval( $ GET['page'] ); 
0 
else 
{ 
Spage = 1; 
L 
// 每 页 数量 
Spage size = 57 
// 获取 总 数据 量 


$sql = "select count(*) from mu"; 
$result = mysql query($sql); 
$row = mysql_ fetch row($result); 
$amount = $row[0]; 
echo "共有 ".$amount." 记 录 &nbsp; gnbsp;"; 
// 计算 共有 多 少 页 
if($amount) 
1 
if( $amount < $page size ) 
{ 
// 如 果 总 数据 量 小 于 page_size， 那 么 只 有 一 页 
Spage_count = 1; 
} 
// 取 总 数据 量 除 以 每 页 数 的 余数 
if( $amount $% Spage size ) 
{ 
// 如 果 有 余数 ， 则 页 数 等 于 总 数据 量 除 以 每 页 数 的 结果 取 整 再 加 一 


Spage_count = (int) ($amount / Spage _ size) + 1; 
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Spage_string .= ' 下 一 页 | 尾 页 '; 
和 


elseti 
Spage_string .= '<a href=?page='. (Spage+1) - '> 下 一 页 </a>1<a href=?page=' . 
$page_count.'> 尾 页 </a>'; 8 
} “0 
echo "<br>".$page string; 
2 
</td> 
</tr> 
</table> 
<table width="100%" border="0" cellspacing="0" cellpadding="0"> 
<tr> 第 
<td background="images/bgd.gif" height="26">&gnbsp;</td> 19 
</tr> ea 
</table> 投 
<p><img src="images/str.gif" width=26 height=27 alt="" hspace="20"> 版 权 所 有 @IT 在 票 
中 国 [www.itzcn.net]</p> 管 
<p>gnbsp;</p> 理 
</BODY> 过 


将 上 述 文件 保存 ， 文 件 名 
为 tou.php。 上 述 代码 中 字体 为 
黑体 的 部 分 主要 是 HTML 代 
码 ， 用 来 实现 该 页 面 的 静态 部 i 
分 ， 这 部 分 代码 会 在 其 他 PHP 
页 面 中 出 现 ， 如 果 再 次 出 现 就 从 当 前 页 而 可 以 查看 投票 信息 
不 再 重复 了 ， 以 省 略 号 代替 。 ee 
中 间 的 PHP 代码 主要 实现 该 模 
块 的 动态 内 容 。 

单 击 图 19-2 中 的 【投票 统 EE 
计 】 超 级 链接 , 会 显示 如 图 19-3 


终 ”版权 所 有 @IT 在 中 国 [www itzcn net] 
所 示 示 的 窗 口 。 | 滞 Mtt> 110:alest 80801toa/tone shy7i 10aL In 中华 环 二 小 姑 评 这 A11:- 总 沈 


在 tou.php 文件 中 , 实现 功 
能 非常 简单 ， 就 是 在 PHP 页 面 
中 以 分 页 形式 显示 项 目 表 mu 中 的 信息 ， 这 样 可 以 在 显示 页 面 上 ， 选 择 自 己 要 查看 的 投票 项 目 。 这 
里 需要 注意 的 是 在 当前 页 面 中 使 用 超级 链接 进行 了 多 个 参数 传递， 其 语句 如 下 所 示 : 


19-3 ”查看 投票 信息 


<a href=tong.php? iid=".$id."g&iin=".$name."g&iis=".$shi.">".$name."</a> 


19.4.2 ”实现 统计 页 面 


如 果 要 查看 选中 项 目的 投票 情况 ， 需 要 获取 投票 信息 表 xin 中 的 数据 。 下 面 实现 投票 的 信息 统 
计 页 面 ， 打 开 记事 本 ， 输 入 下 列 代码 : 


<table width="750" border="0" cellspacing="0" cellpadding="0" class="styles"> 
<tr> 
<td wiqdth="81" valign="top"> 
<img src="images/rc.gif" width=45 height=61 alt=""> 
</td> 
<td width="669"> 
<table border=1 align=center width=59%> 
<tr><td colspan=4></td></tr> 
<tr><td></td><td> 选 项 </td><tq> 比 例 </td><td> 票 数 </td></tr> 
<?php 
$strl=$ GET['iin']; 
$str2=$_GET['iid']; 
$link=mysql connect ("localhost","root", "root"); 
mysql select db("piao"); 
$sql="select * from xin where mu id=$str2"; 
$result=mysql query($sql); 
Snum = mysql num rows ($result); 
echo "本 次 " .$strl." 参 与 投票 的 人 数 为 " .$num; 
$sq="select xuan name,count(3) as piao from xin Where mu id=".$str2." group by 
xuan name order by piao desc limit 0,3"; 
$resultl=mysql query($sq); 
$i=1; 
while($rs=mysql fetch object ($result1)) 
{ 
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Sname=Srs->xuan name; 
$numl=$rs->piao; 
$av= ($numl/$num) *100; 
echo "<tr><td>".$i."</td><td><br>".$name."</td><td>".$av."%<img src= 
images/bg.gif height=10 width=".$av."></td><td>".$numl."</td></tr>"; 
Si++7 
上 
mysql_close(); 
?> 
</td> 
</tr> 
</table> 


将 上 述 代 码 保存 , 文件 名 为 tpng.php。 上 述 代码 中 上 下 两 次 省 略 部 分 和 tou.php 文件 中 黑体 部 分 
完全 相同 。 当 单 击 图 19-3 中 【中 华 环球 小 姐 评选 】 超 级 链接 时 ， 会 显示 如 图 19-4 所 示 的 窗口 。 

从 图 19-3 中 可 以 看 出 该 投票 项 目的 投票 情况 ， 如 参加 投票 人 数 、 获 取 投 票数 前 三 个 选项 。 在 该 
PHP 文件 中 ， 首 先 获取 要 查看 投票 项 目 序列 号 变量 st2， 以 该 变量 作为 参数 获取 投票 信息 表 中 该 项 
目的 所 有 选项 个 数 即 投票 数 。 获 取 完 参加 投票 人 数 之 后 ， 使 用 SQL 语句 “ $sq="select 
xuan name,count(3) as piao from xin where mu id=".$str2." group by xuan name order by piao desc limit 
0,3";” 在 信息 表 中 进行 查询 来 获取 投票 数 最 多 的 前 三 个 选项 ， 并 显示 出 来 。 在 显示 时 ， 使 用 柱状 图 
显示 是 一 个 很 好 的 技巧 ， 希 望 读者 注意 。 
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1 刘海 松 23.5294117647% 一 


2 黄蜂 23.5294117647% 一 


3 刘 红 起 17.6470588235% 二 


新 权 所 有 加 T 在 中 国 [wkzon no 


图 19-4 投票 统计 显示 
19.5 ”投票 管理 模块 


票 管理 模块 是 一 个 只 有 系统 管理 员 才 可 以 登录 的 模块 。 在 本 软件 中 ， 该 模块 的 使 用 没有 用 户 
校 验 部 分 ， 因 为 前 面 多 次 提 到 了 用 户 校 验 的 实现 。 投 票 管理 模块 主要 是 对 投票 项 目 和 相应 选项 的 管 
理 ， 在 该 模块 中 可 以 实现 投票 项 目的 添加 、 删 除 、 修 改 、 查 看 功能 、 选 项 的 删除 和 选项 的 修改 。 单 
击 图 19-2 中 的 【投票 管理 】 超 级 链接 ， 可 以 进入 该 模块 。 


19.5.1 实现 管理 显示 页 面 


下 面 编写 投票 管理 模块 的 显示 页 面 ， 打 开 记事 本 ， 输 入 下 列 内 容 : 


<table width="750" border="0" cellspacing="0" cellpadding="0" class="styles"> 
<tr> 
<td width="81" valign="top"> 
<img src="images/rc.gif" width=49 height=60 alt=""> 
</td> 
<td width="669"> 
<p> 本 页 面 为 管理 员 操作 页 面 ， 可 做 如 下 的 操作 </p> 
<table border=] align=center width=85%> 
<th align=left> 项 目 名 称 </th> 
<th align=left> 删 除 </th> 
<th align=left> 修 改 </th> 
<th align=left> 追 加 选项 </th> 
<th align=left> 选 项 操作 </th> 
<th align=left> 查 看 选项 </th> 
<?php 


$link=mysql connect ("localhost", "root", "root"); 


忆 站 


章 
投 
管 
理 
系 
统 


471 


PHP+MySQL 休 钊 久久 中 认 


将 上 述 代码 保存 , 文件 名 为 muguanl.php。 文 件 中 省 略 的 部 分 同 tou.php 文件 中 的 黑体 部 分 相同 。 
单 击 图 19-2 中 的 【投票 管理 】 超 级 链接 ， 会 显示 如 图 19-5 所 示 的 窗口 。 


ET 
狼 tp :yilocalaost 3000/ wovernl pay 


本 页面 思 关 理 铅 潍 作 页 而 ， 可 名 如 下 当 检 作 


项 目 名 称 
E23 


中 华 环球 小 姐 评选 


台 


[本 ] 居 


这 
< 


C3 
阳光 少女 EE 
最 佳美 少女 GE 

i 
Ca 


最 吸引 女人 奖 
运动 员 


ET 


图 19-5 投票 管理 模块 


在 上 述 代码 中 , 首先 获取 投票 项 目 表 mu 中 所 有 的 投票 项 目 , 并 以 表格 形式 显示 在 PHP 页 面 中 。 
在 显示 时 ,只 显示 投票 项 目 名 称 , 而 投票 项 目 序列 号 以 隐藏 按钮 值 存储 起 来 , 传递 到 下 面 的 页 面 中 ， 
如 下 所 示 : 


<input type=hidden name=xuan value=".$id."> 


在 此 页 面 中 ， 可 以 进行 上 面 的 5 项 操作 ， 如 删除 一 个 投票 项 目 、 修 改 一 个 投票 项 目 等 。 
19.5.2 ”删除 操作 


下 面 创 建 一 个 PHP 页 面 实 现 删 除 投票 项 目 操作 ， 打 开 记 事 本 ， 输 入 下 列 代 码 : 


<table width="750" border="0" cellspacing="0" cellpadding="0" class="styles"> 
<tr> 
<td width="81" valign="top"> 
<img src="images/rc.gif" width=39 height=24 alt=""> 
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</td> 
<td width="669"> 
<?php 
require once "piaoCon.php"; 
$id=$ POST['del']7 
Sp=new Pcon(); 
$sql="delete from mu where mu id=$id"; 
Sp->inS ($sql); 
echo "被 删除 项 目的 序列 号 为 " .$id; 
$sq="delete from xuan where mu id=$id"; 
$p->ins ($sq); 
echo "<script Language='JavaScript'>window.alert (' 项 目 删除 成 功 ， 即 将 返回 项 目 管理 页 
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面 ')</script>"; 
echo "<script Language="'JavaScript'>window.location ='muguan]l .php'</script>"; 
3> 
</td> 
</tr> 
</table> 


将 上 述 代码 保存 , 文件 名 为 mudel php。 省 略 部 分 同 tou.php 文件 中 的 黑体 部 分 相同 , 后 面 的 PHP 


文件 中 如 果 出 现 省 略 号 而 没有 特殊 说 明 ， 和 此 处 含义 一 致 。 当 单 击 图 19-5 中 的 【删除 】 按 钮 ， 会 显 
示 如 图 19-6 所 示 的 窗口 。 


全 二 地 Intrmet 


图 19-6 ”删除 页 面 


在 上 述 代码 中 ,不 但 要 删除 项 目 表 mu 中 关于 该 项 目的 信息 ， 也 要 删除 选项 表 xuan 中 关于 该 项 
目的 选项 。 如 果 投 票 项 目 删除 成 功 则 返回 投票 管理 页 面 。 


19.5.3 ”修改 操作 


修改 操作 可 以 修改 投票 项 目 名 称 ， 如 把 最 佳 男 主持 人 改 为 最 佳 女 主持 人 等 。 
1. 修改 显示 页 面 
下 面 创建 PHP 页 面 实现 该 操作 显示 界面 ， 打 开 记事 本 ， 输 入 下 列 代码 : 


<table width="750"” border="0" cellspacing="0" cellpadding="0" class="styles"> 
<tr> 
<td width="81" valign="top"> 
<img src="images/rc.gif" width=38 height=24 alt=""> 
</td> 
<td width="669"> 
<form action="mugai.php" method=post> 
<?php 
Satr=$ POST[" XI0"]? 
$1link=mysql_ connect ("localhost", "root", "root"); 
mysql select db("piao"); 
$sql="select * from mu where mu id='$str'"; 


$result=mysql query ($sql); 
while($rs=mysql fetch object ($result)) 
{ 


$id=$rs->mu id; 
$name=$rs->mu name; 
echo "输入 要 修改 的 项 目 名 称 :<input type=hidden name=guanxiu value=".$id."><input 
type=text name=xiu Value=" .Sname."><br>"7 . 
. 
} . 
. 
. 
.0 
. 


mysql close(); . 


了 
<input type=submit value= 提 交 修 改 > 
</form> 
</td> 
二 
</table> 


将 上 述 代码 保存 ， 文 件 名 为 muxiu.php。 单 击 图 19-5 中 的 【修改 】 按 钮 ， 会 显示 如 图 19-7 所 示 


[| 


的 窗口 。 
在 至 述 代 码 中 获 取 从 文件 中 编 铝 区) 音 看 0 收 窜 Q) 工具 0 大 屿 00) 


DD 疙 hy /catort 0000/ nen pe 


muguanll.php 页 面 传递 过 来 的 参 
数 ， 即 选项 序列 号 ， 并 以 此 为 参数 不通 软件 公司 
获取 投票 项 目 表 中 mu 该 项 目 名 
称 。 名 称 获取 完毕 后 ， 在 【输入 要 ago 一 
修改 的 项 目 名 称 】 文 本 框 中 显示 ， 


在 此 文本 框 中 可 以 直接 修改 。 
2. 修改 页 面 终 ”版 权 所 有 加 IT 在 中 国 [www itZcn. net] 


在 文本 框 修改 完毕 后 ， 需 要 把 上 EEC 
信息 提交 给 另外 一 个 页 面 进行 处 
理 。 下 面 创建 一 个 PHP 页 面 ， 用 来 

实现 处 理 提交 的 信息 ， 打 开 记 事 本 ， 输 入 下 列 内 容 : 


<table width="750" border="0" cellspacing="0" cellpadding="0" class="styles"> 
<tr> 
<td width="81" valign="top"> 
<img src="images/rc.gif" width=44 height=24 alt=""> 
</td> 
<td width="669"> 
<?php 
require once "piaoCon.php"; 
$str=$ POST['guanxiu’']; 
$strl=$ POST['xiu']; 
if(strlen (trim($strl1))==0){ 
echo "<script Language='JavaScript'>window.alert (' 请 输入 要 修改 的 项 目 ') 
</script>"; 
echo "<script Language="'JavaScript'>window.location ='muguan]l .php'</script>"; 
} 
elsei{ 
Sp=new Pcon(); 
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图 19-7 修改 显示 页 面 
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$sql="update mu set mu name="$strl' where mu id=$str"; 
$p->ins ($sql); 
echo "<script Language='JavaScript'>window.alert(' 项 目 修 改 成 功 ， 即 将 返回 项 目 管理 页 
面 ')</script>"; 
echo "<script Language= "JavaScript'>window.location ='muguan]l .php'</script>"; 
} 
2 
</td> 
</tr> 
</table> 


将 上 述 代码 保存 ， 文 件 名 为 
mugai.php。 单 击 图 19-7 中 的 【提交 
修改 】 按钮， 会 显示 如 图 19-8 所 示 
的 窗口 。 

在 上 述 代码 中 ， 首 先 判断 提交 
信息 是 否 为 室 ， 如 果 为 空 ， 则 将 程 
序 控制 权 转 交 给 项 目 管理 页 面 ， 否 


则 将 会 根据 提交 信息 来 修改 该 项 目 
名 称 。 图 19-8 修改 成 功 页 面 
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19.5.4 ”追加 操作 


在 投票 过 程 中 ， 有 时 需要 中 间 添 加 投票 选项 ， 像 世界 十 大 杰出 人 物 评选 ， 可 以 在 评选 过 程 中 添 
加 一 个 名 称 为 “刘海 松 ” 的 选项 。 追 加 操作 就 是 实现 这 种 功能 的 。 

1. 追加 显示 页 面 

下 面 创 建 一 个 PHP 页 面 ， 实 现 投票 项 目 中 追加 选项 的 显示 页 面 ， 打 开 记 事 本 ， 输 入 下 列 代码 : 


<table width="750" border="0" cellspacing="0" cellpadding="0" class="styles"> 
<tr> 
<td width="81" valign="top"> 
<img src="images/rc.gif" width=34 height=37 alt=""> 
</td> 
<td width="669"> 
<form action="zui.php" method=post> 
选项 名 称 : <input type=text name=name5><br> 
<?php 
require once "piaoCon.php"; 
$id=$_ POST['jia']; 
echo "<input type=hidden name=zui value=".$id.">"; 
<input type=submit value=" 提 交 选 项 "><input type=reset value=" 重 置 "> 
</form> 
</td> 
</tr> 
</table> 
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将 上 述 代码 保存 ， 文 件 名 为 muzuiphp。 在 该 文件 中 使 用 require_once 语句 把 piaoCon.php 文件 
包含 到 当前 文件 中 ， 使 用 隐藏 按钮 
把 要 追加 的 投票 项 目 序列 号 存储 
起 来 。 单 击 图 19-5 中 的 【追加 】 按 
钮 ， 会 显示 如 图 19-9 所 示 的 窗口 。 

在 【选项 名 称 】 文 本 框 中 输入 
信息 后 ， 可 以 把 信息 提交 给 另外 一 
个 页 面 进 行 修改 。 

2. 追加 页 面 

下 面 创 建 一 个 PHP 页 面 , 实现 此 版 权 所 有 回 IT 在 中 国 [www. itZcn net] 
处 理 muzui.php 页 面 提交 信息 。 打 


开 记 事 本 ， 输 入 下 列 代码 ; 图 199 追加 显示 页 面 
<table width="750" border="0" cellspacing="0" cellpadding="0" class="styles"> 
<tr> 


<td width="81" valign="top"> 
<img src="images/rc.gif" width=38 height=24 alt=""> 
</td> 
<td width="669"> 
<?php 
require once "piaoCon.php"; 
$name5=$_POST['name5°']; 
$id=$_POST['zui']; 
if(strlen(trim(Sname5) )==0) 1{ 
echo "<script Language='JavaScript'>window.alert(' 请 输入 要 追加 的 选项 ' ) 
<Jacrint>n? 
echo "<script Language="'JavaScript'>window.location ='muguan]l .php' 
</acript>ny 
} 
elsef{ 
$p=new Pcon(); 
$sql="insert into xuan(mu id,xuan name) values ('$id','$name5')"; 
$p->ins ($sql); 
echo "项 目 添加 成 功 "; 
echo "<script Language='JavaScript'>window.alert(' 项 目 添加 成 功 ， 即 将 返回 项 目 添加 页 
面 ')</script>"; 
echo "<script Language='JavaScript'>window.location ='muguanl .php'</script>"; 
FE 
学 
</td> 
</tr> 
</table> 


将 上 述 代 码 保存 ， 文 件 名 为 zuiphp。 在 该 文件 中 ， 首 先 包含 piaoCon.php 文件 ， 并 获取 要 追加 
票 项 目的 序列 号 。 如 果 文本 框 中 的 提交 信息 为 空 ， 则 返回 投票 管理 页 面 ， 否 则 以 获取 的 序列 号 和 
选项 名 称 为 参数 形成 SQL 语句 ， 然 后 调用 类 Pcon 的 实例 化 对 象 p 的 方法 inSO 执 行 。 如 果 项 目 追 加 


[| 


章 
二 
管 
理 
系 
统 


477 


成 功 ， 则 会 显示 相应 的 信息 。 。 EEC 


单 击 图 19-9 中 的 【提交 选 “| 二 证 ia 
项 】 按 钮 ， 会 显示 如 图 19-10 所 
示 的 窗口 。 


19.5.5 ”选项 操作 


选项 操作 主要 是 针对 投票 
项 目 选项 而 言 的 。 在 该 操作 中 ， 图 19-10 追加 成 功 页 面 
可 以 对 选择 的 投票 项 目 选项 执行 删除 和 修改 操作 。 

1， 选 项 操作 显示 页 面 

下 面 创建 一 个 PHP 页 面 ， 实 现 选 项 操作 的 显示 页 面 。 打 开 记 事 本 ， 输 入 下 列 内 容 


<table width="750" border="0" cellspacing="0" cellpadding="0" class="styles"> 
<tr> 
<td width="81" valign="top"> 
<img src="images/rc.gif" width=42 height=42 alt=""> 
</td> 
<td width="669"> 
<table border=1l1 align=center width=50$%> 
<th> 选 项 名 称 </th><th> 删 除 </th><th> 修 改 </th> 
<?php 
$strl=$ POST[xuan]; 
$str2=$_POST[xuanl]; 
echo "<font color=orange>".$str2."</font>"; 
echo "项 目 具 有 选项 如 下 表 所 示 ， 并 且 能 够 进行 表 中 的 操作 "; 
$1link=mysql_connect ("localhost", "root", "root"); 
mysql_ select db("piao"); 
$sql="select * from xuan Where mu id=$strl"; 
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$result=mysql query($sql); 
while($rs=mysql fetch object ($result)) 
{ 
$id=$rs->xuan id7 
$name=$rs->xuan name; 
echo “<tr> 
<td>" .Sname."</td> 
<td> 
<form action=xuandel .php method=post> 
<input type=hidden name=xdel value=".$id."> 
<input type=submit value= 删 除 > 
</form> 
</td> 
<td> 


<form action=xuanxiu.php method=post> 
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<input type=hidden name=xxiu value=".$id."> 
<input type=hidden name=xxiul value=".$name."> 
<input type=submit value= 修 改 ></form> 
</td> 
< 
} 
mysql close(); 
2 
</table> 
<center><a href=muguanl .php> 返 回 管理 页 面 </a></center> 
</td> 

< Er 

</table> 


将 上 述 代 码 保存 , 文件 名 为 muxuan.php。 单 击 图 19-5 中 的 【选项 操作 】 按 钮 , 会 显示 如 图 19-11 
所 示 的 窗口 。 


Er 
由 站 国境 http://1localhost;3060/to 


惠 通 软件 公司 


但 让 II 日 且 有 进项 加 下 表 所 示 ， 开 且 联 二 表 中 3 栓 作 


选项 名 称 删除 ” 修改 


刘海 松 


图 ”版权 所 有 国 IT 在 中 国 [ww itZen. net] 


图 19-11 选项 显示 页 面 


上 面 的 muxuan.php 文件 主要 实现 将 指定 投票 项 目 选 项 名 称 显示 在 PHP 页 面 ， 并 针对 每 个 投票 
项 目 选项 实现 删除 和 修改 操作 。 

2. 选项 删除 页 面 

下 面 创 建 PHP 页 面 ， 实 现 投票 项 目 选项 删除 操作 ， 打 开 记 事 本 ， 输 入 下 列 代码 ; 


<table width="750"” border="0" cellspacing="0" cellpadding="0" class="styles"> 
EP 
<td width="81" valign="top"> 
<img src="images/rc.gif" width=38 height=24 alt=""> 
</td> 
<td width="669"> 
<?php 
require once "piaoCon.php"; 
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$id=$ POST["xdel’]; 

Sp=new Pcon(); 

$sql="delete from xuan where xuan id=$id"; 

$p->ins ($sql); 

echo "被 删除 项 目的 序列 号 为 " . $iqd; 

echo "<script Language='JavaScript'>window.alert(' 项 目 删除 成 功 ， 即 将 返回 项 目 管理 页 
面 ')</script>"; 

echo "<script Language="'JavaScript'>window.1location ='muguan]l .php'</script>"; 


VD 2> 
</td> 


</tr> 
</table> 


将 上 述 代码 保存 ， 文 件 名 
为 xuandel.php。 单 击 图 19-11 
中 的 【删除 】 按 钮 ， 会 显示 如 
图 19-12 所 示 的 窗口 。 

在 该 文件 中 ， 依 据 从 
muxuan.php 中 获取 的 选项 序列 
号 信息 ， 进 行 删 除 操作 。 

3. 选项 修改 页 面 

下 面 创 建 PHP 页 面 ， 实 现 
对 投票 项 目 选 项 修改 操作 。 打 开 记 事 本 ， 输 入 下 列 代码 : 


<table width="750" border="0" cellspacing="0" cellpadding="0" class="styles"> 
> 
<td width="81"” valign="top"> 
<img src="images/rc.gif" width=55 height=24 alt=""> 
</td> 
<td width="669"> 
<form action=xuanx.php method=post> 
<?php 
$strl=$ POST['xxiu']; 
$str2=$_POST['xxiul']; 
echo $str2." 选 项 改 为 : "7 
echo "<input type=hidden name=x value=".$strl."> 
<input type=hidden name=xx value=".$str2.">"; 
2 
<input type=text name=xxx><br> 
<input type=submit value=" 提 交 修 改 "> 
<form> 
</td> 
</tr> 
</table> 


将 上 述 文件 保存 ， 文 件 名 为 xuanxiu.php。 该 PHP 页 面 为 选项 名 称 修改 显示 页 面 。 单 击 图 19-11 
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图 19-12 选项 删除 页 面 


中 的 【修改 】 按钮 ， 会 显示 如 文件 CJ) 匆 模 四 至 看 CD) 收 芋 册 工具 四。 帮助 CD 
图 19-13 所 示 的 窗口 。 EEC 

在 该 文件 中 ， 主 要 依据 从 
muxuan.php 中 获取 的 选项 序列 号 
信息 ， 进 行 修改 操作 。 数 据 修改 
完毕 后 ， 需 要 提交 给 另外 的 处 理 
页 面 。 下 面 创建 PHP 页 面 ， 实 现 

票 项 目 选项 名 称 修改 操作 。 打 有 多 IT 在 中 itZcn net] 
开 记事 本 ， 输 入 下 列 代码 : 


地 Jntranet 


图 19-13 ”修改 显示 页 面 


<table width="750" border="0" cellspacing="0" cellpadding="0" class="styles"> 东 

<tr> 章 
<td width="81" valign="top"> 

<img src="images/rc.gif" width=32 height=24 alt=""> 投 

</td> 谷 

<td width="669"> 理 

<?php 系 

统 


require once "piaoCon.php"; 

$strl=$_POST[zx]; 

$str2=$ POST[xx]; 

$str3=$_POST[xxx]; 

if(strlen(trim($str3))==0){ 
echo "<script Language='JavaScript'>window.alert(' 请 输入 要 修改 的 选项 名 称 ' ) 
</script> ns 
echo "<script Language='JavaScript'>window.location ='muguan]l .php' 
hecript> ns 

} 

elsef{ 

$p=new Pcon(); 

$sql="update xuan set xuan name='$str3' where xuan id=$strl"; 

echo $sql; 

$p->ins ($sql); 

echo "选项 " .$str2." 被 修改 为 " .$str3; 

echo "<script Language='JavaScript'>window.alert(' 项 目 修改 成 功 ， 即 将 返回 项 目 管理 页 

面 ')</script>"; 

echo "<script Language='JavaScript'>window.location ='muguan]l .php'</script>"; 

} 

?> 
</td> 
</tr> 
</table> 


将 上 述 代码 保存 , 文件 名 为 xuanx.php。 单 击 图 19-13 中 的 【提交 修改 】 按 钮 , 会 显示 如 图 19-14 
所 示 的 窗口 。 
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项 目 相 届 成 功 ， 即 洛 返回 项 下 管 杰 页 面 


19-14 ”选项 修改 成 功 页 面 


19.5.6 ”查看 操作 


卫 
5 
主 该 项 操作 可 以 查看 指定 投票 项 目 中 所 包含 的 选项 ， 并 在 PHP 页 面 中 显示 出 来 。 下 面 创建 一 个 
多 PHP 页 面 来 实现 该 功能 。 打 开 记 事 本 ， 输 入 下 列 代码 : 
加 <table width="750" border="0" cellspacing="0" cellpadding="0" class="styles"> 
3 <tr> 
学 <td width="81" valign="top"> 
<img src="images/rc.gif" width=37 height=24 alt=""> 
册 </td> 

<td width="669"> 

<?php 


$id=$ POsT['cha']; 
$name=$_POST['chal']; 
echo "序列 号 为 " .$id." 的 项 目 <font size=7 color=pear>".$name."</font> 具 有 下 列 选项 : "; 
$1link=mysql_connect ("localhost", "root", "root"); 
mysql_ select db("piao"); 
$sql="select * from xuan where mu id=$id"; 
$result=mysql query ($sql); 
while($rs=mysql fetch object ($result)) 
E 
$name=$rs->xuan name; 
echo "<hr size='1' noshade='noshade' color='pear' style='border-bottom- 
style:dotted' width=80%><center>".S$name."</center>"; 
} 
mysql close(); 
2 
<a href="muguanl .php"> 返 回 管理 页 面 </a> 
</td> 
</tr> 
</table> 


将 上 述 代码 保存 ， 文 件 名 为 mucha.php。 单 击 图 19-5 中 的 【查看 】 按 钮 ， 会 显示 如 图 19-15 所 
示 的 窗口 。 
在 上 述 文件 中 ， 显 示 的 是 在 选项 表 中 指定 项 目 选项 。 
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19.6 ”选项 管理 模块 


选项 管理 模块 主要 完成 投票 项 目 和 选项 添加 操作 ， 如 添加 投票 项 目 、 给 投票 项 目 添加 相应 的 选 
项 。 该 模块 是 系统 管理 员 操作 的 模块 。 单 击 图 19-5 中 的 【选项 管理 】 超 级 链接 会 进入 该 模块 。 


19.6.1 选项 管理 显示 页 面 


下 面 创 建 PHP 页 面 ， 实 现 自动 添加 投票 操作 的 显示 。 打 开 记事 本 ， 输 入 下 列 代 码 ; 


<table width="750" border="0" cellspacing="0" cellpadding="0" class="styles"> 
<tr> 
<td width="81" valign="top"> 
<img src="images/rc.gif" width=35 height=57 alt=""> 
</td> 
<td width="669"> 
<form action=muadd.php method=post> 
项 目 名 称 : <input type=text name=namel><br> 
投票 方式 ，<input type=radio name=name2 value= 单 选 > 单 选 
<input type=radio name=name2 value= 复 选 > 复 选 <br> 
<input type=submit name=name3 value=" 添 加 项 目 "> 
</form> 
</td> 
</tr> 
</table> 


将 上 述 代码 保存 ， 文 件 名 为 muadd0.php。 单 击 图 19-5 中 的 【选项 管理 】 超 级 链接 ， 会 显示 如 
图 19-16 所 示 的 窗口 。 


图 ”版 权 所 有 鲜 IT 在 中 国 [ww itz=n net] 


ET PE 


图 19-16 ”投票 项 目 添加 


在 该 文件 中 ， 主 要 以 表单 形式 显示 了 添加 投票 项 目 操作 的 显示 页 面 ， 这 时 可 以 在 【项 目 名 称 】 
文本 框 中 输入 要 添加 的 项 目 名 称 ， 并 选择 投票 方式 ， 单 选 还 是 复 选 。 


19.6.2 ”投票 项 目 添加 页 面 


单 击 图 19-16 中 的 【添加 项 目 】 按钮, 会 把 当前 页 面 的 信息 提交 给 另外 一 个 处 理 该 信息 的 页 面 。 
下 面 创建 PHP 页 面 ， 实 现 处 理 提交 的 信息 。 打 开 记事 本 ， 输 入 下 列 代码 : 


<?php 

session start(); 

require once "piaoCon.php"; 
3 
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<table width="750" border="0" cellspacing="0" cellpadding="0" class="styles"> 
过 人 
<td width="81" valign="top"> 
<img src="images/rc.gif" width=37 height=24 alt=""> 
</td> 
<td width="669"> 
<?php 
$namel=$_POST['namel']; 
$name2=$_POST['name2']; 
$ SESSION['na']=$namel; 
if(strlen(trim($namel))==0 | strlen(trim(Sname2) )==0) 1{ 
echo "<script Language='JavaScript'>window.alert (' 请 输入 创建 的 项 目 ') 
</script>"y 
echo "<script Language="JavaScript'>window.location ='muadd0 .php' 
</script>"; 
} 
else{ 
S$p=new Pcon(); 
$sql="insert into mu (mu name,mu shi) values ('$namel','S$name2')"; 
$p->ins ($sql); 
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请 耐心 等 待 或 者 刷新 重 试 


<tr><td><input type="submit" value=" 提 交 "></td><td> 
<input type="reset" value=" 重 置 ">></td></tr> 
</form> 

</table> 


单 击 工具 栏 上 的 园 图标 , 保存 上 述 代码 。 单 击 图 21-3 中 的 【产品 录入 】 超 级 链接 ， 会 把 程序 控 
制 权 转 到 ru.php 文件 ， 该 文件 的 执行 结果 如 图 21-4 所 示 。 
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ETT 
图 21-4 产品 录入 显示 页 面 


在 该 图 中 可 以 输入 相应 的 产品 信息 ， 提 交 给 其 他 页 面 进行 处 理 。 单 击 Zend 开发 工具 中 的 【 文 
件 】 代 文件 菜单 】 命 令 ， 将 产生 的 PHP 文件 命名 为 Ih.php， 用 来 处 理 客户 提交 的 产品 信息 。 在 编辑 
器 中 输入 下 列 代码 : 


<?php 

$namel=$_POST['namel']; 

$name2=$_POST['name2°']; 

$name3=$ POST['name3']; 

$name4=$ POST['name4']; 

$name5=$ POST['name5']; 

$name6=$_POST['name6']; 
if(strlen(trim($namel))==0 | strlen(trim(Sname2) )==0 
| strlen (trim(Sname3) )==0 | strlen(trim(Sname4) )==0 
| strlen (trim(Sname5) )==0 | strlen(trim(Sname6) )==0 
lereg("[0-9]",Sname3) ==false |ereg("[0-9]",$name5)==false 
| ereg("[0-9] {4}-[0-9] {1,2}-[0-9] {1,2}", $name4)==false) 
{ 


echo "<script Language='JavaScript'>window.alert (' 请 输入 产品 的 正确 信息 ')</script>"; 
echo "<script Language="'JavaScript'>window.location ='ru.php'</script>"; 
} 


elsel{ 
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Fredquire("piaoCcon.php") 7 

Sname7=Sname3+Sname57 

$sql="insert into ru values('$namel','$name2','$name3','$name4','$name5', 
"Sname7'，'"Sname6') "7 

$sq="insert into cun values('$namel','S$name2','$name5','S$name3','S$name3', 
"Sname6') "7 。 
$p=new Pcon(); 

$p->ins ($sql); 

$p->ins ($sq); 

echo "产品 录入 成 功 "; 

echo "<a href='ru.php'> 继 续 录 入 </a>"; 


} 
2 
单 击 工具 栏 上 的 辕 图 标 ,保存 上 述 代码 。 在 图 21-4 中 输入 产品 的 信息 后 ， 单 击 【 提 交 】 按 钮 ， 
会 显示 如 图 21-5 所 示 的 窗口 。 
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图 21-5 产品 录入 成 功 页 面 


在 该 文件 中 ， 使 用 了 正则 表达 式 用 来 校 验 提交 的 产品 信息 ， 如 果 格 式 正确 ， 就 会 将 数据 插入 进 
货 表 ru 和 库存 表 cun 中 。 


21.4.2 ”产品 进货 查询 


单 击 Zend 开发 工具 中 的 【文件 】|【 打 开 文 件 】 命 令 ， 将 产生 的 PHP 文件 命名 为 jincha.php。 
该 文件 主要 用 于 查询 进货 表 数 据 ， 作 为 参考 依据 ， 其 查询 方式 有 产品 名 称 、 价 格 、 进 货 日 期 。 在 编 
辑 器 中 输入 下 列 代码 : 


<h3> 该 页 面 可 以 完成 多 种 类 型 的 查询 </h3> 

<p> 请 输入 产品 名 称 进行 查询 

<form action="jincha2.php" method=post name=forml> 
<input type=text name=namel><br> 

<input type="submit"” value= 提 交 > 

</form> 

<p> 请 输入 产品 的 价格 进行 查询 (大 于 该 价格 的 产品 ) </p> 

<form action="jincha3.php" method=post name=form2> 
<input type=text name=name2><br> 

<input type="submit" value= 提 交 > 

</form> 
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<p> 请 输入 产品 入 库 日 期 进行 查询 (该 产品 日 期 之 后 ) </p> 

<form action="jincha4.php" method=post name=form3> 
<input type=text name=name3><br> 

<input type="submit"” value= 提 交 > 

</form> 


单 击 工具 栏 上 的 辕 图 标 ,保存 上 述 代码 。 单 击 图 21-3 中 的 【进货 查询 】 超 级 链接 ， 会 把 程序 控 
制 权 转 到 jincha.php 文件 ， 该 文件 的 执行 结果 如 图 21-6 所 示 。 
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图 21-6 进货 查询 显示 页 面 


在 该 图 中 可 以 输入 相应 的 产品 查询 信息 ， 并 提交 给 其 他 页 面 进行 处 理 。 对 于 不 同 的 查询 ， 所 提 
交 的 页 面 并 不 相同 。 

1， 按 产品 名 称 查询 

单 击 Zend 开发 工具 中 的 【文件 】| 【文件 菜单 】 命 令 ， 将 产生 的 PHP 文件 命名 为 jincha2.php， 
用 来 处 理 提交 的 产品 名 称 信息 。 在 编辑 器 中 输入 下 列 代 码 : 


<table border=] width="100%" align="center"> 
<caption> 该 产品 的 进货 信息 </caption> 
<th> 编 号 </th><th> 名 称 </th><th> 数 量 </th><th> 时 间 </th><th> 单 价 </th><th> 总 价 </th><th> 
说 明 </th> 
<?php 
$name=$ POST['namel']; 
$1link=mysql_ connect ("localhost","root", "root"); 
mysql select db("chan™"); 
$sql="select * from ru Where chan name like '%$name%®'™"; 
$result=mysql query ($sql); 
Snum = mysql num rows ($result); 
if($num>0){ 
while($rs=mysql fetch object ($result)) 
{ 
$bian=$rs->chan id; 
$name=$rs->chan name; 


542 
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请 耐心 等 待 或 者 刷新 重 试 


加 载 中 


请 耐心 等 待 或 者 刷新 重 试 


HI 眉 悄 沿 19SAW+dHd 


单 击 工具 栏 上 的 回 图 标 ， 保 存 上 述 代码 。 单 击 图 21-3 中 的 【产品 销售 】 超 级 链接 ， 会 把 程序 控 
制 权 转 到 xiaoshou.php 文件 ， 该 文件 的 执行 结果 如 图 21-10 所 示 。 


加 载 中 


请 耐心 等 待 或 者 刷新 重 试 
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Snum = mysql num rows ($result); 
if ($num>0){ 
while($rs=mysql fetch object ($result)) 
{ 
$bian=$rs->chan id; 
$price=$rs->chan price; 
$xian=$rs->chan xian; 
$zhu=$rs->chan zhu; 
$ SESSION['id']=$bian; 
$_SESSION['price']=$price; 
$_SESSION['pr']=$str2*$price*#$str3; 
$_SESSION['xian']=$xian; 
$_SESSION['zhu']=$zhu; 
if($str2<=$xian){ 
echo "现在 你 购买 数量 为 " .$str2. "价格 合计 为 : " .$str2*$price*$str3." 元 ， 请 付 账 "; 
echo "<form action=xiaoshou2.php method=post><input type=submit value= 
确认 完毕 ></form>"; 
} 
ES 
echo "现在 库存 数量 为 ". $xian."， 不 能 满足 你 的 要 求 ， 请 重新 调整 数量 " ; 
} 
} 
mysql close(); 
} 
elset 


echo "该 产品 已 被 卖 完 "; 


涯 上 的 园 图 标 ， 保 存 上 述 代码 。 在 图 21-10 中 的 每 个 文本 框 中 输入 相应 的 购买 信息 。 
按钮 ， 会 把 程序 控制 权 转 到 xiaoshoul.php 文件 ， 该 文件 的 执行 结果 如 图 21-11 所 示 。 
至 I 在 中 国 (http://wew. itzen. com) 
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图 21-11 购买 情况 显示 页 面 


3. 购买 产品 

在 当前 窗口 中 会 显示 客户 的 购买 信息 ， 如 购买 的 数量 、 此 次 交易 的 总 价 等 。 如 果 客 户 核对 后 ， 
确认 无 误 ， 可 以 把 购买 信息 提交 给 另外 的 页 面 处 理 。 单 击 Zend 开发 工具 中 的 【文件 】|【 文 件 菜单 】 
命令 ， 将 产生 的 PHP 文件 命名 为 xiaoshou2.php， 用 来 处 理 客户 的 购买 信息 。 在 编辑 器 中 输入 下 列 
代码 : 


加 载 中 


请 耐心 等 待 或 者 刷新 重 试 


加 载 中 


请 耐心 等 待 或 者 刷新 重 试 


加 载 中 


请 耐心 等 待 或 者 刷新 重 试 
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<td>" .Sbian."</td> 

<td>".$name."</td> 

<td>".$price."</td> 

<td>".$shu."</td> 

<td>".$kou."</td> 

<td>".$pr."</td> 
<HEPS™ 


} 
echo "本 月 总 的 销售 额 为 <font color=red size=6>".$pricenum."</font> 元 "; 
mysql close(); 
else { 
echo "<script Language='JavaScript'>window.alert (' 今 天 没有 销售 额 ') </script>"; 
echo "<script Language="'JavaScript'>window.location ='index.php'</script>"; 
外 
?> 
</table> 


单 击 工具 栏 上 的 园 图 标 ,保存 上 述 代码 。 单 击 图 21-3 中 的 【月 结算 】 超 级 链接 ， 会 把 程序 控制 
权 转 到 yuejie.php 文件 ， 该 文件 的 执行 结果 如 图 21-14 所 示 。 
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图 21-14 月 结算 显示 页 面 


在 yuejie.php 文件 中 ， 需 要 注意 的 是 “select * from xiao where MONTH(chan date)='$da' AND 
YEAR(chan_date)='$dal ”语句 的 使 用 ， 即 获取 指定 年 份 和 月 份 的 记录 信息 。 


21.6 ”产品 库存 管理 模块 


产品 库存 模块 主要 完成 产品 的 库存 盘点 和 查询 ， 如 查看 产品 的 数量 是 否 充足 ， 查 询 当前 库存 中 
有 哪些 产品 。 该 模块 以 数据 表 cun 为 操作 依据 ， 按 功能 划分 两 个 部 分 ,分别 为 库存 盘点 和 库存 查询 。 


21.6.1 库存 盘点 


单 击 Zend 开发 工具 中 的 【文件 】|【 打 开 文 件 】 命 令 ， 将 产生 的 PHP 文件 命名 为 cundian php。 


加 载 中 


请 耐心 等 待 或 者 刷新 重 试 


PHP+MySQL 也 人 纠 镍 尺 目 昌 


<?php 
// 翻 页 链接 


Spage_string = "7 


if( $page — 1 ){ a 
$page_string -= ' 第 一 页 | 上 一 页 1'; a 
! “0 
elsef 和 
$page string .= '<a href=?page=1> 第 一 页 </a>|<a href=?page='.($page-1) .'> 上 一 页 
</a>l'; 
} 
if( ($page == $page count) || ($page count == 0) ){ 


Spage_string .= ' 下 一 页 | 尾 页 '; 

} 

elsef 
Spage_string .= '<a href=?page='. (Spage+1) . "> 下 一 页 </a>1<a href=?page='.$page_ 
count . "> 尾 页 </a>'7 

} 

echo "<br>".$page string; 

> 


单 击 工具 栏 上 的 辕 图 标 ,保存 上 述 代码 。 单 击 图 21-3 中 的 【库存 盘点 】 超 级 链接 ， 会 把 程序 控 
制 权 转 到 cundian.php 文件 ， 该 文件 的 执行 结果 如 图 21-15 所 示 。 
加 TI 在 中 国 (http://wew. itzen. co 
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EETETET 
图 21-15 ”库存 信息 查看 页 面 


在 该 图 中 ， 如 果 库存 数量 比较 充足 则 显示 绿色 ， 否 则 显示 红色 。 完 成 上 面 这 个 操作 ， 主 要 是 依 
据 库存 的 数量 进行 判断 。 


21.6.2 ”库存 查询 


单 击 Zend 开发 工具 中 的 【文件 】 【打开 文 件 】 命 令 ， 将 产生 的 PHP 文件 命名 为 ruchaphp。 该 
文件 是 库存 查询 显示 页 面 。 在 编辑 器 中 输入 下 列 代码 : 
<p> 请 输入 产品 名 称 进行 查询 


<form action="ruchal.php" method=post name=forml> 
<input type=text name=namel><br> 


加 载 中 


请 耐心 等 待 或 者 刷新 重 试 


echo "<tr> 
<td>".$bian."</td> 
<td>".$name."</td> 
<td>". Sprice."</td> 
<td>".$xian."</td> 
<td>".$shu."</td> . 
<td>".$zhu."</td> 
“Er 
} 
mysql close(); 
4 
else { 
echo "<script Language='dJavaScript'">window.alert(" 要 寻找 的 产品 不 存在 ') </script>"; 
echo "<script Language="'JavaScript'>window.location ='rucha.php'</script>"; 
1 
?> 
</table> 


单 击 工具 栏 上 的 辕 图 标 ， 保 存 上 述 代码 。 在 图 21-16 中 输入 要 查询 的 信息 后 ， 单 击 【 提 交 】 按 
钮 ， 会 显示 如 图 21-17 所 示 的 窗口 。 
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该 产品 的 进货 信息 


编号 名 称 单价 现存 数量 进货 数量 ， 说 明 
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本 二 地 Jntranet， 
图 21-17 进货 查询 页 面 


在 上 述 文件 中 ， 会 依据 输入 的 查询 信息 显示 获取 信息 。 如 果 没 有 查询 信息 ， 则 重新 返回 查询 
页 面 。 


21.7 ”用户 管理 模块 


户 管理 模块 主要 完成 用 户 的 相关 操作 ， 如 登录 系统 、 注 册 新 会 员 、 会 员 资料 修改 、 管 理 用 户 。 
该 模块 以 用 户 表 user 为 操作 依据 ， 按 功能 划分 4 个 部 分 ， 分 别 为 用 户 登录 、 用 户 注册 、 用 户 资料 修 
改 、 用 户 管理 。 


21.7.1 ”用户 登录 


单 击 Zend 开发 工具 中 的 【文件 】| 【打开 文件 】 命 令 ， 将 产生 的 PHP 文件 命名 为 login.php。 该 


文件 是 用 户 进行 登录 的 显示 页 面 。 在 编辑 器 中 输入 下 列 代码 : 


<center> 
<h3> 请 输入 用 户 名 称 和 密码 登录 </h3> 
<form action=loginl.php method=post> 
用 户 名 称 : <input type=text name=usename maxlength="25"><br> 


用 户 密码 : <input type="password" name=passname maxlength="35"><br> 
<input type="submit” value=" 提 交 "><input type=reset value=" 重 置 "> 
</form> 
<a href=zhuce.php> 新 用 户 注册 </a> 
</center> 


单 击 工具 栏 上 的 辕 图 标 ,保存 上 述 代码 。 打开 下 浏览 器 ， 在 地 址 栏 中 输入 http://localhost:8080/ 
chan/login.php， 单 击 【 转 到 】 按 钮 ， 会 显示 如 图 21-18 所 示 的 窗口 。 
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图 21-18 用 户 登 录 页 面 


在 该 图 中 可 以 输入 相应 的 登录 信息 , 提交 给 另外 的 页 面 进行 处 理 。 单 击 Zend 开发 工具 中 的 【 文 
件 】| 【文件 菜单 】 命 令 ， 将 产生 的 PHP 文件 命名 为 login1.php， 用 来 处 理 客户 提交 的 用 户 信 息 。 在 
编辑 器 中 输入 下 列 代 码 : 


<?php 
session start(); 
$name=$_POST['usename']; 
$pass=$_POST['passname']; 
echo $name.$pass; 
if(strlen(trim($name)==0 | strlen(trim($pass)))==0){ 
echo "<script Language='JavaScript'>window.alert (' 请 输入 注册 的 名 称 和 密码 ' ) </script>"; 
echo "<script Language="'JavaScript'>window.location ='login.php'</script>"; 
} 
elsef 
$sql="select * from user where user name='$name' and user password='$pass'"; 
$1link=mysql_connect ("localhost", "root", "root"); 
mysql_ select dbl("chan"); 
echo $sql; 
$result=mysql query($sql); 
Snum = mysql num rows ($result); 
if($num>0){ 
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加 载 中 


请 耐心 等 待 或 者 刷新 重 试 


加 载 中 


请 耐心 等 待 或 者 刷新 重 试 


呈送 


单 击 工具 栏 上 的 加 图 标 ， 保 存 上 述 代码 。 在 图 21-18 中 单 击 【 提 交 】 按 钮 ， 程 序 控制 权 会 转向 
zhuce.php 页 面 ， 如 果 提 交 的 信息 符合 校 验 格 式 ， 则 会 转 到 登录 页 面 login.php 页 面 登录 ， 和 否则 会 重 
新 转 到 注册 页 面 。 其 执行 结果 如 图 21-21 所 示 。 
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图 21-21 用 户 注 册 成 功 


21.7.3 ”用 户 资料 修改 


单 击 Zend 开发 工具 中 的 【文件 】| 【打开 文 件 】 命 令 ， 将 产生 的 PHP 文件 命名 为 xiugai.php。 
该 文 作用 于 获取 用 户 的 信息 并 显示 ， 以 方便 用 户 进行 修改 。 在 编辑 器 中 输入 下 列 代码 : 


<center> 

<form action=xiugail.php method=post> 

<h1> 请 在 文本 域 中 直接 修改 </h1> 

<?php 

$name=$_SESSION['username']; 

$link=mysql connect ("localhost", "root", "root"); 

mysql_select db("chan"); 

$sql="select * from user Where user name="'S$name'"; 

$result=mysql query($sql); 

while($rs=mysql fetch object ($result)) 
Sname=Srs->user name; 
Spass=5Srs->user password; 
echo "姓名 :<input type=text name=namel value=" .Sname ."><br> 

密码 :<input type=text name=name2 value=".$pass."><br>"; 
} 
mysql close(); 

> 站 

<input type="submit" value=" 提 交 "> 

<input type="reset" value=" 重 置 "> 


山 袜 中 


产 
品 
进 
销 
存 
管 
理 
系 
统 


</form> 
</center> 


单 击 工具 栏 上 的 辕 图 标 , 保存 上 述 代码 。 在 图 21-3 中 单 击 【用户 修 改 】 超 级 链接 ， 程 序 控制 权 
会 转向 xiugai.php 页 面 ， 执 行 结果 如 图 21-22 所 示 。 
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图 21-22 ”修改 显示 页 面 


在 该 图 中 可 以 输入 相应 的 修改 信息 , 提交 给 另外 的 页 面 进行 处 理 。 单 击 Zend 开发 工具 中 的 【 文 
件 】| 【文件 菜单 】 命 令 ， 将 产生 的 PHP 文件 命名 为 xiugail.php， 用 来 处 理 客户 提交 的 用 户 修改 信 
息 。 在 编辑 器 中 输入 下 列 代码 : 


<?php 

$na=$_SESSION['username']; 

$name=$ POST['namel']; 

$pass=$ POST['name2']7 

ifl(strlen(trim($name)==0 | strlen(trim($pass)))==0){ 
echo "<script Language='JavaScript'>window.alert (' 请 输入 要 修改 名 称 和 密码 ') </script>"; 
echo "<script Language="'JavaScript'>window.location ='xiugai.php'</script>"; 

. 


else{ 
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require("piaoCcon.php") 7 

$sql="update user set user name='$name' , user password='$pass' where 

user name='$na'"; 

$p=new Pcon(); 

Sp->inS ($sql); 

echo "修改 数据 成 功 "; 

echo "<script Language='JavaScript'>window.alert (' 修 改 数据 成 功 , 请 重新 登录 ') </script>"; 
echo "<script Language="'JavaScript'>window.location ="'login.php'</script>"; 


?> 


单 击 工具 栏 上 的 园 图 标 ， 保 存 上 述 代码 。 在 图 21-22 中 单 击 【 提 交 】 按 钮 ， 程 序 控制 权 会 转向 
xiugail.php 页 面 ， 如 果 提 交 的 信息 修改 成 功 则 会 显示 如 图 21-23 所 示 的 窗口 。 
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图 21-23 用 户 信 息 修改 页 面 


21.7.4 用 户 管理 


$link=mysql connect ("localhost","root","root"); 
mysql select db("chan"); 

// 获取 当前 页 数 

if( isset($ GET['page']) ){ 
$page = intval( $ GET["page'] ); 


第 

单 击 Zend 开发 工具 中 的 【文件 】|【 打 开 文 件 】 命 令 ， 将 产生 的 PHP 文件 命名 为 guan.php。 该 名 
文件 用 于 对 注册 用 户 进行 管理 ， 其 页 面 是 一 个 显示 页 面 。 在 编辑 器 中 输入 下 列 代码 : 可 
<h5> 对 注册 用 户 可 进行 如 下 操作 : </h5> 品 
<table border=] width=72$% align="center"> 进 
<th> 用 户 名 称 </th><th> 操 作 选项 </th><th> 操 作 选 项 </th> 销 
<?php 在 

// 建立 数据 库 连 接 理 

系 

统 


1 
else 
{ 
Spage = 1; 


} 
// 每 页 数量 
$page size = 37 
// 获取 总 数据 量 
$sql = "select count(*) from User"7 
$result = mysql query($sql); 
$row = mysql fetch row(Sresult) 
$amount = $row[0]; 
echo "共有 " .$amount ." 记 录 &gnbsp; gnbsp;"; 
// 计算 共有 多 少 页 
if ($amount) 
| 

if( $amount < Spage size ) 

{ 

// 如 果 总 数据 量 小 于 page_size， 那 么 只 有 一 页 


$page count = 1; 


} 
// 取 总 数据 量 除 以 每 页 数 的 余数 


if( $amount $ $page size ) 
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> 


<?php 
// 翻 页 链接 
S$page string = "7 
if( $page == 1 ){ 。 
$page_string .=' 第 一 页 | 上 一 页 |'; 二 
一 二 
elsef 
Spage_string .= '<a href=?page=1> 第 一 页 </a>1<a href=?page='. (Spage-1) . '> 上 一 页 
</a>l'; 
} 
if( ($page == $page count) || ($page count == 0) ){ 
Spage_string .= ' 下 一 页 | 尾 页 '; 
} 
else{ 
Spage_string .= '<a href=?page='.($page+1). 7T> 下 一 页 </a>1<a href=?page=' . Spage_ 
count . "> 尾 页 </a>'7 
} 
echo "<br>".$page string; 
?> 


单 击 工具 栏 上 的 辕 图 标 ,保存 上 述 代 码 。 在 图 21-3 中 单 击 【管理 用 户 】 超 级 连接 ， 程 序 控制 权 
会 转向 guan.php 页 面 ， 执 行 结 果 如 图 21-24 所 示 。 
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对 广 抽 用 户 可 进行 如 下 操作 : 
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图 21-24 管理 用 户 显示 页 面 


在 该 图 中 ， 高 级 用 户 可 以 删除 普通 用 户 ， 或 者 设置 普通 用 户 为 高 级 用 户 等 操作 。 

1. 删除 页 面 

单 击 Zend 开发 工具 中 的 【文件 】| 【文件 菜单 】 命 令 ， 将 产生 的 PHP 文件 命名 为 delphp， 用 来 
删除 客户 信息 。 在 编辑 器 中 输入 下 列 代 码 : 


<?php 

$name=$ POST['namel']; 

echo $name; 

require ("piaoCon.php"); 

$sql="delete from user where user name='$name'™"; 


567 


Sp=new Pcon(); 

Sp->inS ($sql); 

echo "删除 数据 成 功 "; 

echo "<script Language='JavaScript'>window.alert (' 该 用 户 已 经 被 删除 ') </script>"; 
echo "<script Language="'JavaScript'>window.location ='guan.php'</script>"; 

2 


单 击 工具 栏 上 的 园 图 标 ， 保 存 上 述 代码 。 在 图 21-24 中 单 击 【 删 除 】 按 钮 ， 程 序 控制 权 会 转向 
中 delphp 页 面 ， 如 果 用 户 信息 成 功 则 会 显示 如 图 21-25 所 示 的 对 话 框 。 


图 21-25 用 户 信息 删除 成 功 


2. 设置 权限 页 面 
单 击 Zend 开发 工具 中 的 【文件 】| 【文件 菜单 】 命 令 , 将 产生 的 PHP 文件 命名 为 shephp， 用 来 
设置 用 户 权 限 。 在 编辑 器 中 输入 下 列 代码 : 


<?php 

$name=$_POST['name2']; 

echo $name; 

require ("piaoCon.php"); 

$sql="update user set user xian=']1' where user name='$name'™"; 

echo $sql; 

$p=new Pcon(); 

$p->ins ($sql); 

echo "该 用 户 修改 权限 成 功 "; 

echo "<script Language='JavaScript'>window.alert (' 用 户 修改 权限 成 功 ')</script>"; 
echo "<script Language='JavaScript"'>window.location ='guan.php'</script>"; 
?> 


单 击 工具 栏 上 的 辕 图 标 ， 保 存 上 述 代码 。 在 图 21-24 中 单 击 【设置 权限 】 按 钮 ， 程 序 控制 权 会 
转向 she php 页 面 ， 如 果 用 户 信息 成 功 则 会 显示 如 图 21-26 所 示 的 对 话 框 。 
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图 21-26 权限 修改 成 功 


21.7.5 “联系 我 们 页 面 


单 击 Zend 开发 工具 中 的 【文件 】| 【打开 文件 】 命 令 ， 将 产生 的 PHP 文件 命名 为 dizhiphp。 该 
文件 是 关于 公司 信息 的 显示 页 面 。 代 码 的 详细 情况 可 以 查阅 光盘 。 在 图 21-3 中 单 击 【联系 我 们 】 超 “ 
级 连接 ， 程 序 的 控制 权 会 转向 dizhi.php 页 面 ， 如 果 用 户 信息 成 功 则 会 显示 如 图 21-27 所 示 的 窗口 。 
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图 21-27 联系 我 们 显示 页 面 
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